diff --git a/ChangeLog b/ChangeLog
index 8ec7657..fa9ebcb 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,30 @@
+* Mon Aug 20 2007 Marc Waeckerlin - mrw-c++-3.2.5 (mrw)
+ - New checkinstall.sh, cause checkinstall does not work otherwise
+ - "#if _REENTRANT" needs "defined(...)"
+* Tue Aug 16 2007 Marc Waeckerlin - mrw-c++-3.2.4 (mrw)
+ - Changed specfiles to eilminate build problems
+* Sun Aug 5 2007 Marc Waeckerlin - mrw-c++-3.2.3 (mrw)
+ - Spec-file changed for SuSEBuild.org
+* Sun Aug 5 2007 Marc Waeckerlin - mrw-c++-3.2.2 (mrw)
+ - Added Valgrind files
+ - No redundant checking in makefile for "make release"
+* Sun Aug 5 2007 Marc Waeckerlin - mrw-c++-3.2.1 (mrw)
+ - Compatibility to UNICODE log4cxx 0.9.7
+* Fri Jul 7 2007 Marc Waeckerlin - mrw-c++-3.2.0 (mrw)
+ - Partly ported to SVN version of log4cxx
+ - Added valgrind support with --enable-valgrind
+ - Checked with valgrind and repaired defects:
+ - Mismatching delete in StackTrace
+* Tue Jun 12 2007 Marc Waeckerlin - mrw-c++-3.1.1 (mrw)
+ - Don't catch abort() in autostacktraceXXX.cpp
+* Thu May 31 2007 Marc Waeckerlin - mrw-c++-3.1.0 (mrw)
+ - Bug in string.hpp join: Only join with list of strings was possible.
+* Fri May 25 2007 Marc Waeckerlin - mrw-c++-3.0.0 (mrw)
+ - Function- and method-trace have been renamed from FUNCTION and
+ METHOD to MRW_FUNCTION and MRW_METHOD due to naming conflicts with
+ Qt-Library. This breaks backward compatibility!
+* Sat May 19 2007 Marc Waeckerlin - mrw-c++-2.3.4 (mrw)
+ - also tracing for signals in autostacktraceXXX (-lmrwexclog4cxx)
* Sat Feb 04 2006 Marc Waeckerlin - mrw-c++-2.3.1 (mrw)
- new rpm package mrw-c++-minimal without debug facilities
no dependencies to boost, log4cxx and ltdl libraries
diff --git a/bootstrap.sh b/bootstrap.sh
index 2d637db..91e94a9 100755
--- a/bootstrap.sh
+++ b/bootstrap.sh
@@ -15,5 +15,4 @@
## added file header
##
-test -f makefile && make distclean
aclocal && libtoolize --force && automake -a && autoconf
diff --git a/checkinstall.sh b/checkinstall.sh
new file mode 100755
index 0000000..d00f53e
--- /dev/null
+++ b/checkinstall.sh
@@ -0,0 +1,11 @@
+if [ "$1" = "-h" -o "$1" = "--help" ]; then
+ echo "All parameters will be passed to checkinstall, call as 'root' - e.g.:"
+ echo " ./checkinstall.sh --pkgname=cross-mingw-mrw-c++ --pkgversion=3.2.4 -y"
+ exit 0
+fi
+TMP=/tmp/mrw-c++-checkinstall
+rm -rf $TMP
+DESTDIR=$TMP make install
+cd $TMP
+checkinstall $* cp -r * /
+rm -rf $TMP
diff --git a/configure.in b/configure.in
index 7cc988c..dcad22c 100644
--- a/configure.in
+++ b/configure.in
@@ -13,11 +13,11 @@
# init
AC_INIT([mrw/mrw.hpp.in])
+AC_CANONICAL_SYSTEM
PACKAGENAME=mrw-c++
-m4_define(x_major, 2)
-m4_define(x_minor, 3)
-m4_define(x_least, 1)
-AM_INIT_AUTOMAKE(@PACKAGENAME@, @MAJOR@.@MINOR@.@LEAST@, [marc@waeckerlin.org])
+m4_define(x_major, 3)
+m4_define(x_minor, 2)
+m4_define(x_least, 5)
# copy M4 to shell
MAJOR=x_major
@@ -27,6 +27,8 @@ AC_SUBST(MAJOR)
AC_SUBST(MINOR)
AC_SUBST(LEAST)
+AM_INIT_AUTOMAKE($PACKAGENAME, $MAJOR.$MINOR.$LEAST, [marc@waeckerlin.org])
+
# libtool versioning
LIB_MAJOR=m4_eval(x_major+x_minor)
LIB_MINOR=x_least
@@ -62,9 +64,17 @@ AC_CHECK_HEADER(sys/old_procfs.h, [AM_CPPFLAGS=-D__solaris__])
# libraries
AC_CHECK_HEADER(log4cxx/logger.h, [have_log4cxx=yes])
AC_CHECK_HEADER(ltdl.h, [have_ltdl=yes])
+AC_CHECK_HEADER(regex.h, [have_regex=yes])
+AC_CHECK_HEADERS(unistd.h fcntl.h, [have_exec=yes])
+AC_CHECK_HEADER(dirent.h, [have_dir=yes])
# Arguments
AM_MAINTAINER_MODE
+AC_ARG_ENABLE(valgrind,
+ AS_HELP_STRING([--enable-valgrind],
+ [runs all checks with valgrind enabled]),
+ [have_valgrind="$enableval"], [have_valgrind="no"])
+AM_CONDITIONAL(HAVE_VALGRIND, test "$have_valgrind" = "yes")
AC_ARG_ENABLE(threads,
[ --disable-threads disable multithreading],
[with_threads="$enableval"], [with_threads="yes"])
@@ -85,6 +95,23 @@ AC_ARG_ENABLE(dot,
[ --disable-dot disable dot graphic tools for documentation],
[have_dot="$enableval"])
test "$enableval" = "yes" && HAVE_DOT="YES" || HAVE_DOT="NO";
+AC_ARG_ENABLE(doxygen,
+ [ --disable-doxygen disable build of documentation],
+ [have_doxygen="$enableval"])
+test "$enableval" = "yes" && HAVE_DOXYGEN="YES" || HAVE_DOXYGEN="NO";
+AM_CONDITIONAL(HAVE_DOXYGEN, test "$enableval" = "yes")
+AC_ARG_ENABLE(dir,
+ [ --disable-dir disable POSIX directory access],
+ [have_dir="$enableval"])
+AM_CONDITIONAL(HAVE_DIR, test "$have_dir" = "yes")
+AC_ARG_ENABLE(exec,
+ [ --disable-exec disable UNIX fork and execute with pipes],
+ [have_exec="$enableval"])
+AM_CONDITIONAL(HAVE_EXEC, test "$have_exec" = "yes")
+AC_ARG_ENABLE(regexp,
+ [ --disable-regexp disable use of regular expressions],
+ [have_regex="$enableval"])
+AM_CONDITIONAL(HAVE_REGEXP, test "$have_regex" = "yes")
AC_ARG_ENABLE(log4cxx,
[ --disable-log4cxx disable use of log4cxx library in automated
stack trace],
@@ -164,6 +191,7 @@ AM_PATH_CPPUNIT([1.0.0], [have_cppunit="yes"], [have_cppunit="no"])
# export macros
AC_SUBST(HAVE_DOT)
+AC_SUBST(HAVE_DOXYGEN)
AC_SUBST(THREADS)
AC_SUBST(PACKAGENAME)
AC_SUBST(AM_CXXFLAGS)
@@ -184,6 +212,9 @@ if test "$have_dot" = "no"; then
AC_MSG_WARN([Missing program dot!
- when you rebild documentation with make doc, there are no generated images
- there are precompiled derived files in the distribution]); fi
+if test "$have_doxygen" = "no"; then
+AC_MSG_WARN([Disabled doxygen!
+ - documentation will not be built]); fi
if test "$have_cppunit" = "no"; then
AC_MSG_WARN([Missing cppunit development library!
- you cannot check the library using "make check"
@@ -197,6 +228,15 @@ if test "$with_threads" = "no"; then
AC_MSG_WARN([Multithreading support is disabled!
- use --enable-threads to enable it
- actually, there's only support for automated function trace]); fi
+if test "$have_dir" != "yes"; then
+AC_MSG_WARN([POSIX directory access is disabled!
+ - you need a UNIX/Posix environment if you want to use it]); fi
+if test "$have_exec" != "yes"; then
+AC_MSG_WARN([UNIX fork/exec with pipes will not be used!
+ - you need a UNIX/Posix environment if you want to use it]); fi
+if test "$have_regex" != "yes"; then
+AC_MSG_WARN([GNU regular expression library will not be used!
+ - if you want it, install it]); fi
if test "$have_log4cxx" != "yes"; then
AC_MSG_WARN([Library log4cxx will not be used!
- if you want it, download it from:
diff --git a/log4cxx-0.9.7-3.i386.rpm b/log4cxx-0.9.7-3.i386.rpm
new file mode 100644
index 0000000..d330971
Binary files /dev/null and b/log4cxx-0.9.7-3.i386.rpm differ
diff --git a/makefile.am b/makefile.am
index 96eed0a..80ab1d1 100644
--- a/makefile.am
+++ b/makefile.am
@@ -11,7 +11,7 @@
SUBDIRS = mrw
-EXTRA_DIST = bootstrap.sh
+EXTRA_DIST = bootstrap.sh checkinstall.sh
nobase_include_HEADERS = mrw/arg.hpp mrw/list.hpp \
mrw/unistd.hpp \
@@ -46,26 +46,35 @@ RPMS = /usr/src/packages/RPMS/i586/@PACKAGENAME@-minimal-@MAJOR@.@MINOR@.@LEAST@
release: tag webserver
tag: distcheck
- cvs ci -R .
- cvs tag -FR REL_@PACKAGENAME@-@MAJOR@-@MINOR@-@LEAST@ .
+ svn ci -m "Release @MAJOR@-@MINOR@-@LEAST@ with tag: REL_@PACKAGENAME@-@MAJOR@-@MINOR@-@LEAST@" .
+ svn cp . -m "Release @MAJOR@-@MINOR@-@LEAST@" file:///home/svn/pro/@PACKAGENAME@/tags/REL_@PACKAGENAME@-@MAJOR@-@MINOR@-@LEAST@
rpm: dist
cp @PACKAGENAME@-@MAJOR@.@MINOR@.@LEAST@.tar.gz /usr/src/packages/SOURCES/
rpmbuild -ba --clean @PACKAGENAME@.spec
- rpmbuild -ba --clean @PACKAGENAME@-minimal.spec
+ rpmbuild -bb --clean @PACKAGENAME@-minimal.spec
-webserver: rpm
- ssh root@waeckerlin.org bash -c \
- '"rm -r /home/marc/mrw-c++; mkdir -p /home/marc/mrw-c++"'
- scp doc/html/* \
- doc/latex/@PACKAGENAME@-@MAJOR@.@MINOR@.@LEAST@.pdf \
- @PACKAGENAME@-@MAJOR@.@MINOR@.@LEAST@.tar.gz \
- ${RPMS} \
- log4cxx*.rpm \
- root@waeckerlin.org:/home/marc/mrw-c++/
+webserver: rpm webserver.en webserver.de webserver.zh
+ - rm -rf ~/www/marc/data/doxygen/mrw-cpp
+ mkdir -p ~/www/marc/data/doxygen/mrw-cpp
+ cp doc/html/*.html ~/www/marc/data/doxygen/mrw-cpp/
+ cp doc/latex/@PACKAGENAME@-@MAJOR@.@MINOR@.@LEAST@.pdf \
+ @PACKAGENAME@-@MAJOR@.@MINOR@.@LEAST@.tar.gz \
+ ${RPMS} \
+ log4cxx*.rpm \
+ ~/www/marc/downloads/
+ date +'%s 127.0.0.1 :computer:c_:mrw-c_:index mrw @PACKAGENAME@ version @MAJOR@.@MINOR@.@LEAST@' >> ~/www/marc/data/changes.log
-distclean-local:
+webserver.%:
+ - rm -rf ~/www/marc/data/media/$(@:webserver.%=%)/computer/c_/mrw-c_
+ mkdir -p ~/www/marc/data/media/$(@:webserver.%=%)/computer/c_/mrw-c_
+ cp doc/html/*.png \
+ doc/html/*.gif \
+ ~/www/marc/data/media/$(@:webserver.%=%)/computer/c_/mrw-c_/
+
+maintainer-clean-local:
- find . -name '*~' | xargs rm
+ - rm -r doc
- rm -r autom4te.cache
- rm aclocal.m4 config.guess config.sub configure \
depcomp install-sh ltmain.sh makefile makefile.in \
diff --git a/mrw-c++-minimal.spec.in b/mrw-c++-minimal.spec.in
index e1ec862..6e881cc 100644
--- a/mrw-c++-minimal.spec.in
+++ b/mrw-c++-minimal.spec.in
@@ -10,22 +10,26 @@
##
# rpmbuild -bb --clean @PACKAGENAME@.spec
-Summary: MRW's C++ Class Library, facilities for modern C++ programming
-Name: @PACKAGENAME@-minimal
+Name: @PACKAGENAME@
Version: @MAJOR@.@MINOR@.@LEAST@
Release: 1
License: LGPL
Group: Development/Libraries/C++
+Summary: dummy
+
+%description
+dummy
+
+%package minimal
+Summary: MRW's C++ Class Library (minimal release, no debugging features)
+Group: Development/Libraries/C++
URL: http://marc.waeckerlin.org/mrw-c++/index.html
Source0: @PACKAGENAME@-@MAJOR@.@MINOR@.@LEAST@.tar.gz
BuildRoot: %{_tmppath}/%{name}-%{version}-%{release}-buildroot
Prefix: /usr
-
-Summary: MRW's C++ Class Library (minimal release, no debugging features)
-Group: Development/Libraries/C++
Provides: @PACKAGENAME@
-%description
+%description minimal
@README@
This package contains only the shared libraries required at runtime,
@@ -35,14 +39,22 @@ http://marc.waeckerlin.org/videorekorder/index.html
%prep
%setup -q
-CXXFLAGS="-O3" ./configure --prefix=$RPM_BUILD_ROOT/usr --datadir=$RPM_BUILD_ROOT/usr/share/doc/packages --disable-threads --disable-autofntrace --disable-log4cxx --disable-ltdl --disable-stacktrace
+CXXFLAGS="-O3" ./configure --prefix=/usr \
+ --datadir=/usr/share/doc/packages \
+ --disable-threads --disable-autofntrace \
+ --disable-log4cxx --disable-ltdl --disable-stacktrace \
+ --disable-doxygen --disable-dot \
+%ifarch x86_64
+ --libdir=/usr/lib64
+%else
+ --libdir=/usr/lib
+%endif
%build
make
%install
-make install
-find "$RPM_BUILD_ROOT" -name '*.la' | xargs perl -pi -e 's#'"${RPM_BUILD_ROOT//+/\\+}"'##g'
+DESTDIR=$RPM_BUILD_ROOT make all install
rm -r $RPM_BUILD_ROOT/usr/include/mrw
rmdir $RPM_BUILD_ROOT/usr/include
rm $RPM_BUILD_ROOT/usr/lib/libmrw-mt*
@@ -56,7 +68,7 @@ rm -r $RPM_BUILD_ROOT/usr/share/doc/packages/@PACKAGENAME@/examples
%clean
rm -rf $RPM_BUILD_ROOT
-%files
+%files minimal
%defattr(-,root,root,-)
/usr/lib/libmrw.so
/usr/lib/libmrw.so.@MAJOR@
diff --git a/mrw-c++.spec.in b/mrw-c++.spec.in
index 0c2124d..29ed801 100644
--- a/mrw-c++.spec.in
+++ b/mrw-c++.spec.in
@@ -5,11 +5,12 @@
## $Date$
## $Author$
##
-## @copy © Marc Wäckerlin
+## @copy © Marc Wäckerlin
## @license LGPL, see file COPYING
##
# rpmbuild -bb --clean @PACKAGENAME@.spec
+BuildRequires: gcc-c++ doxygen graphviz cppunit-devel boost-devel log4cxx-devel freefont te_latex ghostscript-library tetex
Summary: MRW's C++ Class Library, facilities for modern C++ programming
Name: @PACKAGENAME@
Version: @MAJOR@.@MINOR@.@LEAST@
@@ -17,7 +18,7 @@ Release: 1
License: LGPL
Group: Development/Libraries/C++
URL: http://marc.waeckerlin.org/mrw-c++/index.html
-Source0: @PACKAGENAME@-@MAJOR@.@MINOR@.@LEAST@.tar.gz
+Source0: %{name}-%{version}.tar.gz
BuildRoot: %{_tmppath}/%{name}-%{version}-%{release}-buildroot
Prefix: /usr
@@ -31,20 +32,46 @@ This package contains only the shared libraries required at runtime.
%prep
%setup -q
-CXXFLAGS="-O3" ./configure --prefix=$RPM_BUILD_ROOT/usr --datadir=$RPM_BUILD_ROOT/usr/share/doc/packages
+CXXFLAGS="-O3" ./configure --prefix=/usr \
+ --datadir=/usr/share/doc/packages \
+ --enable-doxygen \
+%ifarch x86_64
+ --libdir=/usr/lib64
+%else
+ --libdir=/usr/lib
+%endif
%build
make
%install
-make install
-find "$RPM_BUILD_ROOT" -name '*.la' | xargs perl -pi -e 's#'"${RPM_BUILD_ROOT//+/\\+}"'##g'
+DESTDIR=$RPM_BUILD_ROOT make all install
%clean
rm -rf $RPM_BUILD_ROOT
%files
%defattr(-,root,root,-)
+%ifarch x86_64
+/usr/lib64/libmrw.so
+/usr/lib64/libmrw.so.@MAJOR@
+/usr/lib64/libmrw.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwexcstderr.so
+/usr/lib64/libmrwexcstderr.so.@MAJOR@
+/usr/lib64/libmrwexcstderr.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwexclog4cxx.so
+/usr/lib64/libmrwexclog4cxx.so.@MAJOR@
+/usr/lib64/libmrwexclog4cxx.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwautofunctiontracelog4cxx.so
+/usr/lib64/libmrwautofunctiontracelog4cxx.so.@MAJOR@
+/usr/lib64/libmrwautofunctiontracelog4cxx.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwautofunctiontracestdlog.so
+/usr/lib64/libmrwautofunctiontracestdlog.so.@MAJOR@
+/usr/lib64/libmrwautofunctiontracestdlog.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwlog4cxxconfiguration.so
+/usr/lib64/libmrwlog4cxxconfiguration.so.@MAJOR@
+/usr/lib64/libmrwlog4cxxconfiguration.so.@MAJOR@.@MINOR@.@LEAST@
+%else
/usr/lib/libmrw.so
/usr/lib/libmrw.so.@MAJOR@
/usr/lib/libmrw.so.@MAJOR@.@MINOR@.@LEAST@
@@ -63,6 +90,7 @@ rm -rf $RPM_BUILD_ROOT
/usr/lib/libmrwlog4cxxconfiguration.so
/usr/lib/libmrwlog4cxxconfiguration.so.@MAJOR@
/usr/lib/libmrwlog4cxxconfiguration.so.@MAJOR@.@MINOR@.@LEAST@
+%endif
/usr/share/doc/packages/@PACKAGENAME@/COPYING
/usr/share/doc/packages/@PACKAGENAME@/README
/usr/share/doc/packages/@PACKAGENAME@/AUTHORS
@@ -82,6 +110,26 @@ This Package contains thread safe libraries.
%files mt
%defattr(-,root,root,-)
+%ifarch x86_64
+/usr/lib64/libmrw-mt.so
+/usr/lib64/libmrw-mt.so.@MAJOR@
+/usr/lib64/libmrw-mt.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwexcstderr-mt.so
+/usr/lib64/libmrwexcstderr-mt.so.@MAJOR@
+/usr/lib64/libmrwexcstderr-mt.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwexclog4cxx-mt.so
+/usr/lib64/libmrwexclog4cxx-mt.so.@MAJOR@
+/usr/lib64/libmrwexclog4cxx-mt.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwautofunctiontracelog4cxx-mt.so
+/usr/lib64/libmrwautofunctiontracelog4cxx-mt.so.@MAJOR@
+/usr/lib64/libmrwautofunctiontracelog4cxx-mt.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwautofunctiontracestdlog-mt.so
+/usr/lib64/libmrwautofunctiontracestdlog-mt.so.@MAJOR@
+/usr/lib64/libmrwautofunctiontracestdlog-mt.so.@MAJOR@.@MINOR@.@LEAST@
+/usr/lib64/libmrwlog4cxxconfiguration-mt.so
+/usr/lib64/libmrwlog4cxxconfiguration-mt.so.@MAJOR@
+/usr/lib64/libmrwlog4cxxconfiguration-mt.so.@MAJOR@.@MINOR@.@LEAST@
+%else
/usr/lib/libmrw-mt.so
/usr/lib/libmrw-mt.so.@MAJOR@
/usr/lib/libmrw-mt.so.@MAJOR@.@MINOR@.@LEAST@
@@ -100,6 +148,7 @@ This Package contains thread safe libraries.
/usr/lib/libmrwlog4cxxconfiguration-mt.so
/usr/lib/libmrwlog4cxxconfiguration-mt.so.@MAJOR@
/usr/lib/libmrwlog4cxxconfiguration-mt.so.@MAJOR@.@MINOR@.@LEAST@
+%endif
%package devel
Summary: MRW's C++ Class Library (development files)
@@ -115,6 +164,32 @@ This Package contains all files required for developement.
%files devel
%defattr(-,root,root,-)
+%ifarch x86_64
+/usr/lib64/libmrw.a
+/usr/lib64/libmrw.la
+/usr/lib64/libmrwexcstderr.a
+/usr/lib64/libmrwexcstderr.la
+/usr/lib64/libmrwexclog4cxx.a
+/usr/lib64/libmrwexclog4cxx.la
+/usr/lib64/libmrw-mt.a
+/usr/lib64/libmrw-mt.la
+/usr/lib64/libmrwautofunctiontracelog4cxx-mt.a
+/usr/lib64/libmrwautofunctiontracelog4cxx-mt.la
+/usr/lib64/libmrwautofunctiontracelog4cxx.a
+/usr/lib64/libmrwautofunctiontracelog4cxx.la
+/usr/lib64/libmrwautofunctiontracestdlog-mt.a
+/usr/lib64/libmrwautofunctiontracestdlog-mt.la
+/usr/lib64/libmrwautofunctiontracestdlog.a
+/usr/lib64/libmrwautofunctiontracestdlog.la
+/usr/lib64/libmrwlog4cxxconfiguration.a
+/usr/lib64/libmrwlog4cxxconfiguration.la
+/usr/lib64/libmrwexclog4cxx-mt.a
+/usr/lib64/libmrwexclog4cxx-mt.la
+/usr/lib64/libmrwexcstderr-mt.a
+/usr/lib64/libmrwexcstderr-mt.la
+/usr/lib64/libmrwlog4cxxconfiguration-mt.a
+/usr/lib64/libmrwlog4cxxconfiguration-mt.la
+%else
/usr/lib/libmrw.a
/usr/lib/libmrw.la
/usr/lib/libmrwexcstderr.a
@@ -139,6 +214,7 @@ This Package contains all files required for developement.
/usr/lib/libmrwexcstderr-mt.la
/usr/lib/libmrwlog4cxxconfiguration-mt.a
/usr/lib/libmrwlog4cxxconfiguration-mt.la
+%endif
/usr/include
/usr/share/doc/packages/@PACKAGENAME@/examples
/usr/share/doc/packages/@PACKAGENAME@/doc
diff --git a/mrw/auto.hpp b/mrw/auto.hpp
index eff4660..f894e60 100644
--- a/mrw/auto.hpp
+++ b/mrw/auto.hpp
@@ -16,6 +16,7 @@
#include // munmap, PROT_READ, MAP_SHARED
#include // close
#include
+#include
// forward
class bfd;
@@ -254,6 +255,96 @@ namespace mrw {
T* _res; ///< the resource to be managed
};
+ /** @brief Automatically deletes an array pointer when destructed.
+ @pre \#include
+
+ mrw::AutoPtrAry is a replacement for std::auto_ptr. Other than
+ mrw::AutoPtr it is for arrays, that means, it calls @c delete[]
+ instead of @c delete to free the memory. The problem with
+ standard std::auto_ptr is, that it cannot be stored in a
+ std::map and it does not call @c delete[].
+
+ @warning Use this class with prudence! Should I ever find out,
+ how to work around the std::auto_ptr / std::map problem,
+ then this class may become deprecated.
+
+ @param T type of the pointer to manage
+ */
+ template
+ class AutoPtrAry {
+ public:
+ /// @brief Construct from an allocated resource.
+ /// The resource is freed if necessary.
+ /// AutoPtrAry takes over ownership of the resource.
+ explicit AutoPtrAry(T* res = 0)
+ throw(std::bad_exception): _res(res) {
+ res = 0;
+ }
+ /// @brief Takeover ownership from another AutoPtrAry.
+ AutoPtrAry(AutoPtrAry& o) throw(std::bad_exception):
+ _res(o.release()) {
+ }
+ //! @brief Do not use this method!
+ //! This method had to be introduced for the @c std::map, where
+ //! @c operator[] first creates an empty element, then assigns it
+ //! the real value. Because that empty element is temporary, gcc can
+ //! not use @c AutoResource(AutoResource&) since release 4.0.
+ AutoPtrAry(const AutoPtrAry& o) throw(std::bad_exception): _res(0) {
+ assert(o._res==0);
+ if (o._res!=0)
+ _res = const_cast(o).release();
+ }
+ /// @brief Free resource. Calls @c reset().
+ ~AutoPtrAry() throw(std::bad_exception) {reset();}
+ /// @brief Assign new resource. Calls @c reset().
+ /// The resource is freed if necessary.
+ AutoPtrAry& operator=(T* res) throw(std::bad_exception) {
+ return reset(res);
+ }
+ /// @brief Takeover ownership from another AutoResource.
+ /// Calls @c reset() from @c this and @c release() from @c other.
+ AutoPtrAry& operator=(AutoPtrAry& other) throw(std::bad_exception) {
+ return reset(other.release());
+ }
+ /// @brief Get the resource.
+ operator T* const() const throw(std::bad_exception) {
+ return _res;
+ }
+ /// @brief find out, if a value is set
+ /// @return @c true: resource is valid
+ operator bool() const throw(std::bad_exception) {
+ return _res!=0;
+ }
+ /// @brief Access the AutoPtrAry like a normal pointer.
+ T*const operator->() {return _res;}
+ /// @brief Dereference the AutoPtrAry like a normal pointer.
+ T& operator*() {return *_res;}
+ /// @brief get the resetted resource for resetting it.
+ /// Calls @c reset and returns the cleaned resource.
+ /// The intention is, that you can safely assign it a new value
+ /// (e.g. in an expression).
+ T* getClean() throw(std::bad_exception) {
+ reset();
+ return _res;
+ }
+ /// @brief Give away ownership of the resource.
+ /// @return old resource
+ T* release() throw(std::bad_exception) {
+ T* res(_res); _res=0;
+ return res;
+ }
+ /// @brief Assign a new resource.
+ /// The old resource of @c this is freed if necessary.
+ AutoPtrAry& reset(T* res = 0)
+ throw(std::bad_exception) {
+ delete[] _res;
+ _res = res;
+ return *this;
+ }
+ private:
+ T* _res; ///< the resource to be managed
+ };
+
/** @brief Automatically closes a file when destructed.
@pre \#include
diff --git a/mrw/autofunctiontracelog4cxx.cpp b/mrw/autofunctiontracelog4cxx.cpp
index db5568f..b3e77c2 100644
--- a/mrw/autofunctiontracelog4cxx.cpp
+++ b/mrw/autofunctiontracelog4cxx.cpp
@@ -23,7 +23,8 @@
#include
#include
-#if (__GNUC__==3 && __GNUC_MINOR__<4 || __GNUC__<3) && _REENTRANT && !_MT
+#if (__GNUC__==3 && __GNUC_MINOR__<4 || __GNUC__<3) \
+ && defined(_REENTRANT) && !defined(_MT)
#define _MT
#endif
@@ -31,6 +32,26 @@
extern "C" void __cyg_profile_func_enter (void *, void *) __attribute__((no_instrument_function));
extern "C" void __cyg_profile_func_exit (void *, void *) __attribute__((no_instrument_function));
+
+// support for log4cxx 0.9.7 and for new CVS version
+#ifndef LOG4CXX_LOCATION
+# define MRW_LOG4CXX_LOCATION pos.file.c_str(), pos.line
+#define MRW_LEVEL_DEBUG ::log4cxx::Level::DEBUG
+#define MRW_LEVEL_INFO ::log4cxx::Level::INFO
+#define MRW_LEVEL_WARN ::log4cxx::Level::WARN
+#define MRW_LEVEL_ERROR ::log4cxx::Level::ERROR
+#define MRW_LEVEL_FATAL ::log4cxx::Level::FATAL
+#else
+# define LOG4CXX_CVS
+# define MRW_LOG4CXX_LOCATION \
+ ::log4cxx::spi::LocationInfo(pos.file.c_str(), pos.function.c_str(), pos.line)
+#define MRW_LEVEL_DEBUG ::log4cxx::Level::getDebug()
+#define MRW_LEVEL_INFO ::log4cxx::Level::getInfo()
+#define MRW_LEVEL_WARN ::log4cxx::Level::getWarn()
+#define MRW_LEVEL_ERROR ::log4cxx::Level::getError()
+#define MRW_LEVEL_FATAL ::log4cxx::Level::getFatal()
+#endif
+
namespace mrw {
struct ThreadInfo {
@@ -176,14 +197,20 @@ extern "C" void __cyg_profile_func_enter(void *this_fn, void*) {
(hierarchy.rfind(' ', std::min(hierarchy.find('<'),
hierarchy.find("operator"))));
if (p!=std::string::npos) hierarchy.erase(0, p+1);
+// hierarchy.erase
+// (hierarchy.find_first_not_of
+// ("qwertzuiopasdfghjklyxcvbnmQWERTZUIOPASDFGHJKLYXCVBNM1234567890_."));
+ if (hierarchy=="") hierarchy = "global";
+ std::string name("mrw.fn."+hierarchy);
log4cxx::Logger* logger
- (log4cxx::Logger::getLogger(_T("mrw.fn.")+hierarchy));
+ (log4cxx::Logger::getLogger(log4cxx::String(name.begin(), name.end())));
if (logger->isDebugEnabled()) {
- std::ostringstream ss;
- ss<forcedLog(::log4cxx::Level::DEBUG, ss.str(),
- pos.file.c_str(), pos.line);
+ std::basic_ostringstream ss;
+ ss<forcedLog(MRW_LEVEL_DEBUG, ss.str(),
+ MRW_LOG4CXX_LOCATION);
}
}
} catch (...) {}
@@ -206,19 +233,25 @@ extern "C" void __cyg_profile_func_exit(void *this_fn, void*) {
for (std::string::size_type p(0);
(p=hierarchy.find("::", p))!=std::string::npos;
hierarchy.replace(p, 2, "."));
- hierarchy.erase(hierarchy.find('('));
+ hierarchy.erase(hierarchy.rfind('('));
std::string::size_type p
(hierarchy.rfind(' ', std::min(hierarchy.find('<'),
hierarchy.find("operator"))));
if (p!=std::string::npos) hierarchy.erase(0, p+1);
+// hierarchy.erase
+// (hierarchy.find_first_not_of
+// ("qwertzuiopasdfghjklyxcvbnmQWERTZUIOPASDFGHJKLYXCVBNM1234567890_."));
+ if (hierarchy=="") hierarchy = "global";
+ std::string name("mrw.fn."+hierarchy);
log4cxx::Logger* logger
- (log4cxx::Logger::getLogger(_T("mrw.fn.")+hierarchy));
+ (log4cxx::Logger::getLogger(log4cxx::String(name.begin(), name.end())));
if (logger->isDebugEnabled()) {
- std::ostringstream ss;
- ss<forcedLog(::log4cxx::Level::DEBUG, ss.str(),
- pos.file.c_str(), pos.line);
+ std::basic_ostringstream ss;
+ ss<forcedLog(MRW_LEVEL_DEBUG, ss.str(),
+ MRW_LOG4CXX_LOCATION);
}
}
} catch (...) {}
diff --git a/mrw/autofunctiontracestdlog.cpp b/mrw/autofunctiontracestdlog.cpp
new file mode 100644
index 0000000..ad35965
--- /dev/null
+++ b/mrw/autofunctiontracestdlog.cpp
@@ -0,0 +1,133 @@
+/** @file
+
+ $Id: autofunctiontracelog4cxx.cpp,v 1.3 2005/04/14 19:12:18 marc Exp $
+
+ $Date: 2005/04/14 19:12:18 $
+ $Author: marc $
+
+ @copy © Marc Wäckerlin
+ @license LGPL, see file COPYING
+
+ 1 2 3 4 5 6 7 8
+ 5678901234567890123456789012345678901234567890123456789012345678901234567890
+*/
+
+#ifndef __GNUG__
+#error GNU C++ Compiler is required for automatical function trace
+#endif
+
+#include
+#include
+#include
+#include
+
+#if (__GNUC__==3 && __GNUC_MINOR__<4 || __GNUC__<3) \
+ && defined(_REENTRANT) && !defined(_MT)
+#define _MT
+#endif
+
+// these are special built in functions of GNU Compiler Collection
+extern "C" void __cyg_profile_func_enter (void *, void *) __attribute__((no_instrument_function));
+extern "C" void __cyg_profile_func_exit (void *, void *) __attribute__((no_instrument_function));
+
+namespace mrw {
+
+ // workaround doxygen problem:
+ // - the C++ compiler compiles the following code
+ // - doxygen ignores it
+#ifndef LET_DOXYGEN_IGNORE_THIS // no matching class member found for
+ // int mrw::ThreadInfo::level()
+ struct ThreadInfo {
+ int level;
+ bool recurse;
+ };
+#endif
+
+#ifdef _MT
+ static __thread ThreadInfo info = {0, false};
+#else
+ static ThreadInfo info = {0, false};
+#endif
+
+ static bool mainPassed(false);
+
+ class Lock {
+ public:
+ Lock() __attribute__((no_instrument_function));
+ ~Lock() __attribute__((no_instrument_function));
+ };
+ // workaround doxygen problem:
+ // - the C++ compiler compiles the following code
+ // - doxygen ignores it
+#ifndef LET_DOXYGEN_IGNORE_THIS // doxygen can't match with __attribute__ above
+ Lock::Lock() {
+ info.recurse = true;
+ }
+ Lock::~Lock() {
+ info.recurse = false;
+ }
+#endif
+}
+
+extern "C" int main(int, char**);
+
+/** @addtogroup FunctionTrace */
+//@{
+
+/** @defgroup AutoFunctionTraceStdlog Automatic Function Trace to standard out for GNU g++
+
+ Same as @ref AutoFunctionTrace, but traces to @c stdlog.
+
+ If you link to the library @c libmrwautofunctiontracestdlog using
+ a linker option such as: @c -lmrwautofunctiontracestdlog and you
+ must enable the GNU Compiler Collection specific function trace
+ feature with compile and link option @c -finstrument-functions
+ then you get an automatical function trace, that traces to @c
+ stdlog. You don't need to change a single line in your code!
+
+ */
+//@{
+
+//@}
+
+//@}
+
+extern "C" void __cyg_profile_func_enter(void *this_fn, void*) {
+ if (!mrw::mainPassed)
+ if (this_fn == (void*)&::main) // not ANSI C++ conform...
+ mrw::mainPassed=true;
+ else
+ return;
+ try {
+ if (mrw::info.recurse) return;
+ mrw::Lock lock;
+ {
+ static bool init(mrw::StackTrace::createSymtable());
+ if (!init) return;
+ mrw::StackTrace::CodePos pos(mrw::StackTrace::translate(this_fn));
+ std::clog<
#include
+#include
#include
#include
-#ifndef _T
-#define _T
+
+// support for log4cxx 0.9.7 and for new CVS version
+#ifndef LOG4CXX_LOCATION
+# define MRW_LOG4CXX_LOCATION __FILE__, __LINE__
+#define MRW_LEVEL_DEBUG ::log4cxx::Level::DEBUG
+#define MRW_LEVEL_INFO ::log4cxx::Level::INFO
+#define MRW_LEVEL_WARN ::log4cxx::Level::WARN
+#define MRW_LEVEL_ERROR ::log4cxx::Level::ERROR
+#define MRW_LEVEL_FATAL ::log4cxx::Level::FATAL
+#else
+# define LOG4CXX_CVS
+# define MRW_LOG4CXX_LOCATION \
+ ::log4cxx::spi::LocationInfo(__FILE__, __FUNCTION__, __LINE__)
+#define MRW_LEVEL_DEBUG ::log4cxx::Level::getDebug()
+#define MRW_LEVEL_INFO ::log4cxx::Level::getInfo()
+#define MRW_LEVEL_WARN ::log4cxx::Level::getWarn()
+#define MRW_LEVEL_ERROR ::log4cxx::Level::getError()
+#define MRW_LEVEL_FATAL ::log4cxx::Level::getFatal()
#endif
namespace mrw {
@@ -27,8 +44,10 @@ namespace mrw {
If you link to the library @c libmrwexclog4cxx using a linker
option such as: @c -lmrwexclog4cxx, then an unexpected and a
terminate handler are registered, that trace a fatal error using
- the log4cxx library. You don't need to change a single line in
- your code!
+ the log4cxx library. Also for all know signals there's a signal
+ handler to trace the stack before termination (except in case of
+ a @c SIGTRAP, wehere the program is not terminated). You don't
+ need to change a single line in your code!
The log4cxx library is located at:
- http://logging.apache.org/log4cxx
@@ -49,7 +68,7 @@ namespace mrw {
@code
void unexpected_log4cxx() {
- log4cxx::LoggerPtr logger(log4cxx::Logger::getLogger(_T("mrw.stacktrace")));
+ log4cxx::LoggerPtr logger(log4cxx::Logger::getLogger("mrw.stacktrace"));
logger->fatal("Unexpected Exception", __FILE__, __LINE__);
StackTrace::createSymtable();
std::string st((std::string)StackTrace());
@@ -71,21 +90,27 @@ namespace mrw {
*/
void unexpected_log4cxx() {
- log4cxx::LoggerPtr logger(log4cxx::Logger::getLogger(_T("mrw.stacktrace")));
- logger->fatal("Unexpected Exception", __FILE__, __LINE__);
+ static const std::string name("mrw.stacktrace");
+ log4cxx::LoggerPtr logger
+ (log4cxx::Logger::getLogger(log4cxx::String(name.begin(), name.end())));
+ static const std::string txt("Unexpected Exception");
+ logger->fatal(log4cxx::String(txt.begin(), txt.end()), MRW_LOG4CXX_LOCATION);
StackTrace::createSymtable();
std::string st((std::string)StackTrace());
try {
throw;
} catch (const mrw::exception& x) {
- logger->fatal(std::string("Reason:\n")+x.what()
- +"\nStack:\n"+x.stacktrace());
+ std::string txt(std::string("Reason:\n")+x.what()
+ +"\nStack:\n"+x.stacktrace());
+ logger->fatal(log4cxx::String(txt.begin(), txt.end()));
} catch (const std::exception& x) {
- logger->fatal(std::string("Reason:\n")+x.what()
- +"\nStack:\n"+st);
+ std::string txt(std::string("Reason:\n")+x.what()
+ +"\nStack:\n"+st);
+ logger->fatal(log4cxx::String(txt.begin(), txt.end()));
} catch (...) {
- logger->fatal(std::string("Reason: **** not available ****")
- +"\nStack:\n"+st);
+ std::string txt(std::string("Reason: **** not available ****")
+ +"\nStack:\n"+st);
+ logger->fatal(log4cxx::String(txt.begin(), txt.end()));
}
throw std::bad_exception();
}
@@ -98,7 +123,7 @@ namespace mrw {
@code
void terminate_log4cxx() {
- log4cxx::LoggerPtr logger(log4cxx::Logger::getLogger(_T("mrw.stacktrace")));
+ log4cxx::LoggerPtr logger(log4cxx::Logger::getLogger("mrw.stacktrace"));
logger->fatal("Uncaught Exception", __FILE__, __LINE__);
StackTrace::createSymtable();
std::string st((std::string)StackTrace());
@@ -121,35 +146,104 @@ namespace mrw {
*/
void terminate_log4cxx() {
- log4cxx::LoggerPtr logger(log4cxx::Logger::getLogger(_T("mrw.stacktrace")));
- logger->fatal("Uncaught Exception", __FILE__, __LINE__);
+ static const std::string name("mrw.stacktrace");
+ log4cxx::LoggerPtr logger
+ (log4cxx::Logger::getLogger(log4cxx::String(name.begin(), name.end())));
+ static const std::string txt("Uncaught Exception");
+ logger->fatal(log4cxx::String(txt.begin(), txt.end()), MRW_LOG4CXX_LOCATION);
StackTrace::createSymtable();
std::string st((std::string)StackTrace());
try {
throw;
} catch (const mrw::exception& x) {
- logger->fatal(std::string("Reason:\n")+x.what()
- +"\nStack:\n"+x.stacktrace());
+ std::string txt(std::string("Reason:\n")+x.what()
+ +"\nStack:\n"+x.stacktrace());
+ logger->fatal(log4cxx::String(txt.begin(), txt.end()));
} catch (const std::exception& x) {
- logger->fatal(std::string("Reason:\n")+x.what()
- +"\nStack:\n"+st);
+ std::string txt(std::string("Reason:\n")+x.what()
+ +"\nStack:\n"+st);
+ logger->fatal(log4cxx::String(txt.begin(), txt.end()));
} catch (...) {
- logger->fatal(std::string("Reason: **** not available ****")
- +"\nStack:\n"+st);
+ std::string txt(std::string("Reason: **** not available ****")
+ +"\nStack:\n"+st);
+ logger->fatal(log4cxx::String(txt.begin(), txt.end()));
}
exit(1);
}
+
+ /** @brief segmentation-fault handler, that traces using log4cxx
- //@}
+ The segmentation-fault handler is installed automatically when you link
+ to @c -lmrwexclog4cxx. The implementation of this handler is as follows:
+
+ @code
+ void signal_log4cxx(int sig) {
+ std::string txt;
+ bool abort(true);
+ switch sig {
+ case SIGFPE: txt="SIGFPE: Arithmetic error."; break;
+ case SIGILL: txt="SIGILL: Illegal instruction."; break;
+ case SIGSEGV: txt="SIGSEGV: Segmentation fault"
+ " (invalid access to valid memory)."; break;
+ case SIGBUS: txt="SIGBUS: Invalid pointer dereference"
+ " (access to an invalid memory address)."; break;
+ case SIGABRT: txt="SIGABRT: Funktion 'abort' was called."; break;
+ case SIGIOT: txt="SIGIOT: PDP-11 'iot' instruction"; break;
+ case SIGTRAP: txt="SIGTRAP: Breakpoint instruction."; abort=false; break;
+ case SIGEMT: txt="SIGEMT: Emulator trap."; break;
+ case SIGSYS: txt="SIGSYS: Bad system call."; break;
+ default: txt="Unknown Signal Nr. "+sig; break;
+ }
+ StackTrace::createSymtable();
+ std::string st((std::string)StackTrace());
+ log4cxx::LoggerPtr logger(log4cxx::Logger::getLogger("mrw.stacktrace"));
+ logger->fatal("Aborted by signal: "+txt+"\nStack:\n"+st);
+ if (abort) exit(1);
+ }
+ @endcode
+
+ */
+ void signal_log4cxx(int sig) {
+ std::string txt;
+ bool abort(true);
+ switch (sig) {
+ case SIGFPE: txt="SIGFPE: Arithmetic error."; break;
+ case SIGILL: txt="SIGILL: Illegal instruction."; break;
+ case SIGSEGV: txt="SIGSEGV: Segmentation fault"
+ " (invalid access to valid memory)."; break;
+ case SIGBUS: txt="SIGBUS: Invalid pointer dereference"
+ " (access to an invalid memory address)."; break;
+ case SIGABRT: txt="SIGABRT: Funktion 'abort' was called."; break;
+ case SIGTRAP: txt="SIGTRAP: Breakpoint instruction."; abort=false; break;
+ case SIGSYS: txt="SIGSYS: Bad system call."; break;
+ default: txt="Unknown Signal Nr. "+sig; break;
+ }
+ StackTrace::createSymtable();
+ std::string st((std::string)StackTrace());
+ static const std::string name("mrw.stacktrace");
+ log4cxx::LoggerPtr logger
+ (log4cxx::Logger::getLogger(log4cxx::String(name.begin(), name.end())));
+ std::string errtxt("Aborted by signal: "+txt+"\nStack:\n"+st);
+ logger->fatal(log4cxx::String(errtxt.begin(), errtxt.end()));
+ if (abort) exit(1);
+ }
class AutoStackTrace {
public:
AutoStackTrace() {
std::set_unexpected(&mrw::unexpected_log4cxx);
std::set_terminate(&mrw::terminate_log4cxx);
+ std::signal(SIGFPE, &mrw::signal_log4cxx);
+ std::signal(SIGILL, &mrw::signal_log4cxx);
+ std::signal(SIGSEGV, &mrw::signal_log4cxx);
+ std::signal(SIGBUS, &mrw::signal_log4cxx);
+ std::signal(SIGTRAP, &mrw::signal_log4cxx);
+ std::signal(SIGSYS, &mrw::signal_log4cxx);
}
};
+ //@}
+
// initialize stack traces (load symbols)
static AutoStackTrace _autoStackTrace;
diff --git a/mrw/autostacktracestderr.cpp b/mrw/autostacktracestderr.cpp
index c41bcff..938a4e5 100644
--- a/mrw/autostacktracestderr.cpp
+++ b/mrw/autostacktracestderr.cpp
@@ -11,6 +11,7 @@
*/
#include
#include
+#include
#include
#include
@@ -48,8 +49,10 @@ namespace mrw {
If you link to the library @c libmrwexcstderr using a linker
option such as: @c -lmrwexcstderr, then an unexpected and a
terminate handler are registered, that trace to the standard
- error stream @c std::cerr. You don't need to change a single
- line in your code!
+ error stream @c std::cerr. Also for all know signals there's a
+ signal handler to trace the stack before termination (except in
+ case of a @c SIGTRAP, wehere the program is not terminated). You
+ don't need to change a single line in your code!
*/
//@{
@@ -163,6 +166,67 @@ namespace mrw {
exit(1);
}
+ /** @brief segmentation-fault handler, that traces using stderr
+
+ The segmentation-fault handler is installed automatically when you link
+ to @c -lmrwexcstderr. The implementation of this handler is as follows:
+
+ @code
+ void signal_stderr(int sig) {
+ std::cerr<<"SIGNAL RECEIVED: -------------------------------"<
#include
+#include
#include
#include
-#ifndef __GLIBC__
-#include
-#endif
-mrw::Errno::Errno() throw(): _errno(errno) {}
+mrw::Errno::Errno() throw(): _errnoxxx(errno) {}
mrw::Errno::operator std::string() const throw(std::bad_exception) {
-#ifdef __GLIBC__
- char error[1024];
- return !strerror_r(_errno, error, 1024) ?
- error : std::string("errno=")+_errno;
-#else
- return mrw::ifelse(strerror(_errno), (std::string("errno=")+_errno).c_str());
-#endif
+ return *this;
}
std::string mrw::Errno::string() const throw(std::bad_exception) {
-#ifdef __GLIBC__
+ char* pos(0);
char error[1024];
- return !strerror_r(_errno, error, 1024) ?
- error : std::string("errno=")+_errno;
+#ifdef __GLIBC__
+# if defined __USE_XOPEN2K && !defined __USE_GNU
+ return strerror_r(errno=_errnoxxx, error, 1024)!=-1
+ ? error+std::string(" errno=")+_errnoxxx
+ : std::string("errno=")+_errnoxxx;
+# else
+ return (pos=strerror_r(errno=_errnoxxx, error, 1024))
+ ? pos+std::string(" errno=")+_errnoxxx
+ : std::string("errno=")+_errnoxxx;
+# endif
#else
- return mrw::ifelse(strerror(_errno), (std::string("errno=")+_errno).c_str());
+# if defined __USE_XOPEN2K && !defined __USE_GNU
+ return strerror_r(errno=_errnoxxx, error, 1024)!=-1
+ ? error+std::string(" errno=")+_errnoxxx
+ : std::string("errno=")+_errnoxxx;
+# else
+ return (pos=strerror(errno=_errnoxxx))
+ ? error+std::string(" errno=")+_errnoxxx
+ : std::string("errno=")+_errnoxxx;
+# endif
#endif
}
diff --git a/mrw/errno.hpp b/mrw/errno.hpp
index 12ab30e..23c1cb4 100644
--- a/mrw/errno.hpp
+++ b/mrw/errno.hpp
@@ -60,18 +60,19 @@ namespace mrw {
/// returns the @c errno stored in the constructor
operator int() const throw() {
- return _errno;
+ return _errnoxxx;
}
/// returns the @c errno stored in the constructor
int numerical() const throw() {
- return _errno;
+ return _errnoxxx;
}
//.............................................................. variables
private:
- int _errno;
+ int _errnoxxx; // "xxx" for MinGW-compatibility (there's a
+ // function named _errno)
};
diff --git a/mrw/exec_test.cpp b/mrw/exec_test.cpp
index a0130c8..f1edd42 100644
--- a/mrw/exec_test.cpp
+++ b/mrw/exec_test.cpp
@@ -48,27 +48,40 @@
#include
#include
+#include
+#ifdef __GNUG__
+#define LOG std::clog<<__PRETTY_FUNCTION__<<'@'<<__FILE__<<':'<<__LINE__;
+#else
+#define LOG std::clog<<__FUNCTION__<<'@'<<__FILE__<<':'<<__LINE__;
+#endif
+
class ExecTest: public CppUnit::TestFixture {
public:
void lsTest() {
+ LOG;
std::string res = (mrw::Cmd("/bin/ls"), "-l",
std::string(getenv("srcdir"))+"/..").execute();
CPPUNIT_ASSERT(res.find("COPYING")Last Change: $date
-
+ Last Build: $date