Commit afd586a5 authored by 's avatar

Initial windows support. Now we don't have the stacktrace and several unittests.


git-svn-id: https://google-glog.googlecode.com/svn/trunk@23 eb4d4688-79bd-11dd-afb4-1d65580434c0
parent c54c7356
...@@ -7,66 +7,6 @@ Copyright (C) 1994, 1995, 1996, 1999, 2000, 2001, 2002, 2004, 2005, ...@@ -7,66 +7,6 @@ Copyright (C) 1994, 1995, 1996, 1999, 2000, 2001, 2002, 2004, 2005,
This file is free documentation; the Free Software Foundation gives This file is free documentation; the Free Software Foundation gives
unlimited permission to copy, distribute and modify it. unlimited permission to copy, distribute and modify it.
Glog-Specific Install Notes
================================
*** NOTE FOR 64-BIT LINUX SYSTEMS
The glibc built-in stack-unwinder on 64-bit systems has some problems
with the glog libraries. (In particular, if you are using
InstallFailureSignalHandler(), the signal may be raised in the middle
of malloc, holding some malloc-related locks when they invoke the
stack unwinder. The built-in stack unwinder may call malloc
recursively, which may require the thread to acquire a lock it already
holds: deadlock.)
For that reason, if you use a 64-bit system and you need
InstallFailureSignalHandler(), we strongly recommend you install
libunwind before trying to configure or install google glog.
libunwind can be found at
http://download.savannah.nongnu.org/releases/libunwind/libunwind-snap-070410.tar.gz
Even if you already have libunwind installed, you will probably still
need to install from the snapshot to get the latest version.
CAUTION: if you install libunwind from the URL above, be aware that
you may have trouble if you try to statically link your binary with
glog: that is, if you link with 'gcc -static -lgcc_eh ...'. This
is because both libunwind and libgcc implement the same C++ exception
handling APIs, but they implement them differently on some platforms.
This is not likely to be a problem on ia64, but may be on x86-64.
Also, if you link binaries statically, make sure that you add
-Wl,--eh-frame-hdr to your linker options. This is required so that
libunwind can find the information generated by the compiler required
for stack unwinding.
Using -static is rare, though, so unless you know this will affect you
it probably won't.
If you cannot or do not wish to install libunwind, you can still try
to use two kinds of stack-unwinder: 1. glibc built-in stack-unwinder
and 2. frame pointer based stack-unwinder.
1. As we already mentioned, glibc's unwinder has a deadlock issue.
However, if you don't use InstallFailureSignalHandler() or you don't
worry about the rare possibilities of deadlocks, you can use this
stack-unwinder. If you specify no options and libunwind isn't
detected on your system, the configure script chooses this unwinder by
default.
2. The frame pointer based stack unwinder requires that your
application, the glog library, and system libraries like libc, all be
compiled with a frame pointer. This is *not* the default for x86-64.
If you are on x86-64 system, know that you have a set of system
libraries with frame-pointers enabled, and compile all your
applications with -fno-omit-frame-pointer, then you can enable the
frame pointer based stack unwinder by passing the
--enable-frame-pointers flag to configure.
Basic Installation Basic Installation
================== ==================
......
...@@ -28,7 +28,11 @@ endif ...@@ -28,7 +28,11 @@ endif
glogincludedir = $(includedir)/glog glogincludedir = $(includedir)/glog
## The .h files you want to install (that is, .h files that people ## The .h files you want to install (that is, .h files that people
## who install this package can include in their own applications.) ## who install this package can include in their own applications.)
gloginclude_HEADERS = src/glog/log_severity.h src/glog/logging.h src/glog/raw_logging.h src/glog/vlog_is_on.h src/glog/stl_logging.h ## We have to include both the .h and .h.in forms. The latter we
## put in noinst_HEADERS.
gloginclude_HEADERS = src/glog/log_severity.h
nodist_gloginclude_HEADERS = src/glog/logging.h src/glog/raw_logging.h src/glog/vlog_is_on.h src/glog/stl_logging.h
noinst_HEADERS = src/glog/logging.h.in src/glog/raw_logging.h.in src/glog/vlog_is_on.h.in src/glog/stl_logging.h.in
docdir = $(prefix)/share/doc/$(PACKAGE)-$(VERSION) docdir = $(prefix)/share/doc/$(PACKAGE)-$(VERSION)
## This is for HTML and other documentation you want to install. ## This is for HTML and other documentation you want to install.
...@@ -56,8 +60,10 @@ noinst_SCRIPTS = ...@@ -56,8 +60,10 @@ noinst_SCRIPTS =
TEST_BINARIES = TEST_BINARIES =
TESTS += logging_unittest TESTS += logging_unittest
logging_unittest_SOURCES = $(gloginclude_HEADERS) src/config.h \ logging_unittest_SOURCES = $(gloginclude_HEADERS) \
src/logging_unittest.cc src/logging_unittest.cc \
src/config_for_unittests.h
nodist_logging_unittest_SOURCES = $(nodist_gloginclude_HEADERS)
logging_unittest_CXXFLAGS = $(PTHREAD_CFLAGS) logging_unittest_CXXFLAGS = $(PTHREAD_CFLAGS)
logging_unittest_LDFLAGS = $(PTHREAD_CFLAGS) logging_unittest_LDFLAGS = $(PTHREAD_CFLAGS)
logging_unittest_LDADD = libglog.la $(COMMON_LIBS) logging_unittest_LDADD = libglog.la $(COMMON_LIBS)
...@@ -81,64 +87,73 @@ signalhandler_unittest_sh: signalhandler_unittest ...@@ -81,64 +87,73 @@ signalhandler_unittest_sh: signalhandler_unittest
$(top_srcdir)/src/signalhandler_unittest.sh $(top_srcdir)/src/signalhandler_unittest.sh
TEST_BINARIES += logging_striptest0 TEST_BINARIES += logging_striptest0
logging_striptest0_SOURCES = $(gloginclude_HEADERS) src/config.h \ logging_striptest0_SOURCES = $(gloginclude_HEADERS) \
src/logging_striptest_main.cc src/logging_striptest_main.cc
nodist_logging_striptest0_SOURCES = $(nodist_gloginclude_HEADERS)
logging_striptest0_CXXFLAGS = $(PTHREAD_CFLAGS) logging_striptest0_CXXFLAGS = $(PTHREAD_CFLAGS)
logging_striptest0_LDFLAGS = $(PTHREAD_CFLAGS) logging_striptest0_LDFLAGS = $(PTHREAD_CFLAGS)
logging_striptest0_LDADD = libglog.la $(COMMON_LIBS) logging_striptest0_LDADD = libglog.la $(COMMON_LIBS)
TEST_BINARIES += logging_striptest2 TEST_BINARIES += logging_striptest2
logging_striptest2_SOURCES = $(gloginclude_HEADERS) src/config.h \ logging_striptest2_SOURCES = $(gloginclude_HEADERS) \
src/logging_striptest2.cc src/logging_striptest2.cc
nodist_logging_striptest2_SOURCES = $(nodist_gloginclude_HEADERS)
logging_striptest2_CXXFLAGS = $(PTHREAD_CFLAGS) logging_striptest2_CXXFLAGS = $(PTHREAD_CFLAGS)
logging_striptest2_LDFLAGS = $(PTHREAD_CFLAGS) logging_striptest2_LDFLAGS = $(PTHREAD_CFLAGS)
logging_striptest2_LDADD = libglog.la $(COMMON_LIBS) logging_striptest2_LDADD = libglog.la $(COMMON_LIBS)
TEST_BINARIES += logging_striptest10 TEST_BINARIES += logging_striptest10
logging_striptest10_SOURCES = $(gloginclude_HEADERS) src/config.h \ logging_striptest10_SOURCES = $(gloginclude_HEADERS) \
src/logging_striptest10.cc src/logging_striptest10.cc
nodist_logging_striptest10_SOURCES = $(nodist_gloginclude_HEADERS)
logging_striptest10_CXXFLAGS = $(PTHREAD_CFLAGS) logging_striptest10_CXXFLAGS = $(PTHREAD_CFLAGS)
logging_striptest10_LDFLAGS = $(PTHREAD_CFLAGS) logging_striptest10_LDFLAGS = $(PTHREAD_CFLAGS)
logging_striptest10_LDADD = libglog.la $(COMMON_LIBS) logging_striptest10_LDADD = libglog.la $(COMMON_LIBS)
TESTS += demangle_unittest TESTS += demangle_unittest
demangle_unittest_SOURCES = $(gloginclude_HEADERS) src/config.h \ demangle_unittest_SOURCES = $(gloginclude_HEADERS) \
src/demangle_unittest.cc src/demangle_unittest.cc
nodist_demangle_unittest_SOURCES = $(nodist_gloginclude_HEADERS)
demangle_unittest_CXXFLAGS = $(PTHREAD_CFLAGS) demangle_unittest_CXXFLAGS = $(PTHREAD_CFLAGS)
demangle_unittest_LDFLAGS = $(PTHREAD_CFLAGS) demangle_unittest_LDFLAGS = $(PTHREAD_CFLAGS)
demangle_unittest_LDADD = libglog.la $(COMMON_LIBS) demangle_unittest_LDADD = libglog.la $(COMMON_LIBS)
TESTS += stacktrace_unittest TESTS += stacktrace_unittest
stacktrace_unittest_SOURCES = $(gloginclude_HEADERS) src/config.h \ stacktrace_unittest_SOURCES = $(gloginclude_HEADERS) \
src/stacktrace_unittest.cc src/stacktrace_unittest.cc
nodist_stacktrace_unittest_SOURCES = $(nodist_gloginclude_HEADERS)
stacktrace_unittest_CXXFLAGS = $(PTHREAD_CFLAGS) stacktrace_unittest_CXXFLAGS = $(PTHREAD_CFLAGS)
stacktrace_unittest_LDFLAGS = $(PTHREAD_CFLAGS) stacktrace_unittest_LDFLAGS = $(PTHREAD_CFLAGS)
stacktrace_unittest_LDADD = libglog.la $(COMMON_LIBS) stacktrace_unittest_LDADD = libglog.la $(COMMON_LIBS)
TESTS += symbolize_unittest TESTS += symbolize_unittest
symbolize_unittest_SOURCES = $(gloginclude_HEADERS) src/config.h \ symbolize_unittest_SOURCES = $(gloginclude_HEADERS) \
src/symbolize_unittest.cc src/symbolize_unittest.cc
nodist_symbolize_unittest_SOURCES = $(nodist_gloginclude_HEADERS)
symbolize_unittest_CXXFLAGS = $(PTHREAD_CFLAGS) symbolize_unittest_CXXFLAGS = $(PTHREAD_CFLAGS)
symbolize_unittest_LDFLAGS = $(PTHREAD_CFLAGS) symbolize_unittest_LDFLAGS = $(PTHREAD_CFLAGS)
symbolize_unittest_LDADD = libglog.la $(COMMON_LIBS) symbolize_unittest_LDADD = libglog.la $(COMMON_LIBS)
TESTS += stl_logging_unittest TESTS += stl_logging_unittest
stl_logging_unittest_SOURCES = $(gloginclude_HEADERS) src/config.h \ stl_logging_unittest_SOURCES = $(gloginclude_HEADERS) \
src/stl_logging_unittest.cc src/stl_logging_unittest.cc
nodist_stl_logging_unittest_SOURCES = $(nodist_gloginclude_HEADERS)
stl_logging_unittest_CXXFLAGS = $(PTHREAD_CFLAGS) stl_logging_unittest_CXXFLAGS = $(PTHREAD_CFLAGS)
stl_logging_unittest_LDFLAGS = $(PTHREAD_CFLAGS) stl_logging_unittest_LDFLAGS = $(PTHREAD_CFLAGS)
stl_logging_unittest_LDADD = libglog.la $(COMMON_LIBS) stl_logging_unittest_LDADD = libglog.la $(COMMON_LIBS)
TEST_BINARIES += signalhandler_unittest TEST_BINARIES += signalhandler_unittest
signalhandler_unittest_SOURCES = $(gloginclude_HEADERS) src/config.h \ signalhandler_unittest_SOURCES = $(gloginclude_HEADERS) \
src/signalhandler_unittest.cc src/signalhandler_unittest.cc
nodist_signalhandler_unittest_SOURCES = $(nodist_gloginclude_HEADERS)
signalhandler_unittest_CXXFLAGS = $(PTHREAD_CFLAGS) signalhandler_unittest_CXXFLAGS = $(PTHREAD_CFLAGS)
signalhandler_unittest_LDFLAGS = $(PTHREAD_CFLAGS) signalhandler_unittest_LDFLAGS = $(PTHREAD_CFLAGS)
signalhandler_unittest_LDADD = libglog.la $(COMMON_LIBS) signalhandler_unittest_LDADD = libglog.la $(COMMON_LIBS)
TESTS += utilities_unittest TESTS += utilities_unittest
utilities_unittest_SOURCES = $(gloginclude_HEADERS) src/config.h \ utilities_unittest_SOURCES = $(gloginclude_HEADERS) \
src/utilities_unittest.cc src/utilities_unittest.cc
nodist_utilities_unittest_SOURCES = $(nodist_gloginclude_HEADERS)
utilities_unittest_CXXFLAGS = $(PTHREAD_CFLAGS) utilities_unittest_CXXFLAGS = $(PTHREAD_CFLAGS)
utilities_unittest_LDFLAGS = $(PTHREAD_CFLAGS) utilities_unittest_LDFLAGS = $(PTHREAD_CFLAGS)
utilities_unittest_LDADD = libglog.la $(COMMON_LIBS) utilities_unittest_LDADD = libglog.la $(COMMON_LIBS)
...@@ -146,7 +161,7 @@ utilities_unittest_LDADD = libglog.la $(COMMON_LIBS) ...@@ -146,7 +161,7 @@ utilities_unittest_LDADD = libglog.la $(COMMON_LIBS)
## vvvv RULES TO MAKE THE LIBRARIES, BINARIES, AND UNITTESTS ## vvvv RULES TO MAKE THE LIBRARIES, BINARIES, AND UNITTESTS
lib_LTLIBRARIES += libglog.la lib_LTLIBRARIES += libglog.la
libglog_la_SOURCES = $(gloginclude_HEADERS) src/config.h \ libglog_la_SOURCES = $(gloginclude_HEADERS) \
src/logging.cc src/raw_logging.cc src/vlog_is_on.cc \ src/logging.cc src/raw_logging.cc src/vlog_is_on.cc \
src/utilities.cc src/utilities.h \ src/utilities.cc src/utilities.h \
src/demangle.cc src/demangle.h \ src/demangle.cc src/demangle.h \
...@@ -160,11 +175,17 @@ libglog_la_SOURCES = $(gloginclude_HEADERS) src/config.h \ ...@@ -160,11 +175,17 @@ libglog_la_SOURCES = $(gloginclude_HEADERS) src/config.h \
src/signalhandler.cc \ src/signalhandler.cc \
src/base/mutex.h src/base/googleinit.h \ src/base/mutex.h src/base/googleinit.h \
src/base/commandlineflags.h src/googletest.h src/base/commandlineflags.h src/googletest.h
nodist_libglog_la_SOURCES = $(nodist_gloginclude_HEADERS)
libglog_la_CXXFLAGS = $(PTRHEAD_CFLAGS) -DNDEBUG libglog_la_CXXFLAGS = $(PTRHEAD_CFLAGS) -DNDEBUG
libglog_la_LDFLAGS = $(PTRHEAD_CFLAGS) libglog_la_LDFLAGS = $(PTRHEAD_CFLAGS)
libglog_la_LIBADD = $(COMMON_LIBS) libglog_la_LIBADD = $(COMMON_LIBS)
## The location of the windows project file for each binary we make
WINDOWS_PROJECTS = google-glog.sln
WINDOWS_PROJECTS += vsprojects/libglog/libglog.vcproj
WINDOWS_PROJECTS += vsprojects/logging_unittest/logging_unittest.vcproj
## ^^^^ END OF RULES TO MAKE THE LIBRARIES, BINARIES, AND UNITTESTS ## ^^^^ END OF RULES TO MAKE THE LIBRARIES, BINARIES, AND UNITTESTS
...@@ -178,13 +199,19 @@ rpm: dist-gzip packages/rpm.sh packages/rpm/rpm.spec ...@@ -178,13 +199,19 @@ rpm: dist-gzip packages/rpm.sh packages/rpm/rpm.spec
deb: dist-gzip packages/deb.sh packages/deb/* deb: dist-gzip packages/deb.sh packages/deb/*
@cd packages && ./deb.sh ${PACKAGE} ${VERSION} @cd packages && ./deb.sh ${PACKAGE} ${VERSION}
# TODO(hamaji): We don't support Visual Studio for now. # Windows wants write permission to .vcproj files and maybe even sln files.
## Windows wants write permission to .vcproj files and maybe even sln files. dist-hook:
#dist-hook: test -e "$(distdir)/vsprojects" \
# test -e "$(distdir)/vsprojects" \ && chmod -R u+w $(distdir)/*.sln $(distdir)/vsprojects/
# && chmod -R u+w $(distdir)/*.sln $(distdir)/vsprojects/
libtool: $(LIBTOOL_DEPS) libtool: $(LIBTOOL_DEPS)
$(SHELL) ./config.status --recheck $(SHELL) ./config.status --recheck
EXTRA_DIST = packages/rpm.sh packages/rpm/rpm.spec packages/deb.sh packages/deb \ EXTRA_DIST = packages/rpm.sh packages/rpm/rpm.spec packages/deb.sh packages/deb \
$(SCRIPTS) src/logging_unittest.err src/demangle_unittest.txt $(SCRIPTS) src/logging_unittest.err src/demangle_unittest.txt \
src/windows/config.h src/windows/port.h src/windows/port.cc \
src/windows/preprocess.sh \
src/windows/glog/log_severity.h src/windows/glog/logging.h \
src/windows/glog/raw_logging.h src/windows/glog/stl_logging.h \
src/windows/glog/vlog_is_on.h \
$(WINDOWS_PROJECTS)
This diff is collapsed.
This diff is collapsed.
...@@ -28,13 +28,19 @@ AC_HEADER_STDC ...@@ -28,13 +28,19 @@ AC_HEADER_STDC
AC_CHECK_HEADER(stdint.h, ac_cv_have_stdint_h=1, ac_cv_have_stdint_h=0) AC_CHECK_HEADER(stdint.h, ac_cv_have_stdint_h=1, ac_cv_have_stdint_h=0)
AC_CHECK_HEADER(sys/types.h, ac_cv_have_systypes_h=1, ac_cv_have_systypes_h=0) AC_CHECK_HEADER(sys/types.h, ac_cv_have_systypes_h=1, ac_cv_have_systypes_h=0)
AC_CHECK_HEADER(inttypes.h, ac_cv_have_inttypes_h=1, ac_cv_have_inttypes_h=0) AC_CHECK_HEADER(inttypes.h, ac_cv_have_inttypes_h=1, ac_cv_have_inttypes_h=0)
AC_CHECK_HEADERS(unistd.h, ac_cv_have_unistd_h=1, ac_cv_have_unistd_h=0)
AC_CHECK_HEADERS(syscall.h) AC_CHECK_HEADERS(syscall.h)
AC_CHECK_HEADERS(sys/syscall.h) AC_CHECK_HEADERS(sys/syscall.h)
# For backtrace with glibc. # For backtrace with glibc.
AC_CHECK_HEADERS(execinfo.h) AC_CHECK_HEADERS(execinfo.h)
# For backtrace with libunwind. # For backtrace with libunwind.
AC_CHECK_HEADERS(libunwind.h, ac_cv_have_libunwind_h=1, ac_cv_have_libunwind_h=0) AC_CHECK_HEADERS(libunwind.h)
AC_CHECK_HEADERS(ucontext.h) AC_CHECK_HEADERS(ucontext.h)
AC_CHECK_HEADERS(sys/utsname.h)
AC_CHECK_HEADERS(pwd.h)
AC_CHECK_HEADERS(syslog.h)
AC_CHECK_HEADERS(sys/time.h)
AC_CHECK_HEADERS(glob.h)
AC_CHECK_SIZEOF(void *) AC_CHECK_SIZEOF(void *)
...@@ -50,6 +56,9 @@ AC_CHECK_FUNC(sigaltstack, ...@@ -50,6 +56,9 @@ AC_CHECK_FUNC(sigaltstack,
AC_CHECK_FUNC(dladdr, AC_CHECK_FUNC(dladdr,
AC_DEFINE(HAVE_DLADDR, 1, AC_DEFINE(HAVE_DLADDR, 1,
[Define if you have the `dladdr' function])) [Define if you have the `dladdr' function]))
AC_CHECK_FUNC(fcntl,
AC_DEFINE(HAVE_FCNTL, 1,
[Define if you have the `fcntl' function]))
AX_C___ATTRIBUTE__ AX_C___ATTRIBUTE__
# We only care about these two attributes. # We only care about these two attributes.
...@@ -168,6 +177,7 @@ AC_SUBST(ac_cv_have___builtin_expect) ...@@ -168,6 +177,7 @@ AC_SUBST(ac_cv_have___builtin_expect)
AC_SUBST(ac_cv_have_stdint_h) AC_SUBST(ac_cv_have_stdint_h)
AC_SUBST(ac_cv_have_systypes_h) AC_SUBST(ac_cv_have_systypes_h)
AC_SUBST(ac_cv_have_inttypes_h) AC_SUBST(ac_cv_have_inttypes_h)
AC_SUBST(ac_cv_have_unistd_h)
AC_SUBST(ac_cv_have_uint16_t) AC_SUBST(ac_cv_have_uint16_t)
AC_SUBST(ac_cv_have_u_int16_t) AC_SUBST(ac_cv_have_u_int16_t)
AC_SUBST(ac_cv_have___uint16) AC_SUBST(ac_cv_have___uint16)
......

Microsoft Visual Studio Solution File, Format Version 10.00
# Visual C++ Express 2008
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "libglog", "vsprojects\libglog\libglog.vcproj", "{34BD04BD-BC1D-4BFC-AAFC-ED02D9E960F1}"
EndProject
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "logging_unittest", "vsprojects\logging_unittest\logging_unittest.vcproj", "{DD0690AA-5E09-46B5-83FD-4B28604CABA8}"
ProjectSection(ProjectDependencies) = postProject
{34BD04BD-BC1D-4BFC-AAFC-ED02D9E960F1} = {34BD04BD-BC1D-4BFC-AAFC-ED02D9E960F1}
EndProjectSection
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Win32 = Debug|Win32
Release|Win32 = Release|Win32
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{34BD04BD-BC1D-4BFC-AAFC-ED02D9E960F1}.Debug|Win32.ActiveCfg = Debug|Win32
{34BD04BD-BC1D-4BFC-AAFC-ED02D9E960F1}.Debug|Win32.Build.0 = Debug|Win32
{34BD04BD-BC1D-4BFC-AAFC-ED02D9E960F1}.Release|Win32.ActiveCfg = Release|Win32
{34BD04BD-BC1D-4BFC-AAFC-ED02D9E960F1}.Release|Win32.Build.0 = Release|Win32
{DD0690AA-5E09-46B5-83FD-4B28604CABA8}.Debug|Win32.ActiveCfg = Debug|Win32
{DD0690AA-5E09-46B5-83FD-4B28604CABA8}.Debug|Win32.Build.0 = Debug|Win32
{DD0690AA-5E09-46B5-83FD-4B28604CABA8}.Release|Win32.ActiveCfg = Release|Win32
{DD0690AA-5E09-46B5-83FD-4B28604CABA8}.Release|Win32.Build.0 = Release|Win32
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
EndGlobal
...@@ -63,12 +63,12 @@ ...@@ -63,12 +63,12 @@
#define DECLARE_VARIABLE(type, name, tn) \ #define DECLARE_VARIABLE(type, name, tn) \
namespace FLAG__namespace_do_not_use_directly_use_DECLARE_##tn##_instead { \ namespace FLAG__namespace_do_not_use_directly_use_DECLARE_##tn##_instead { \
extern type FLAGS_##name; \ extern GOOGLE_GLOG_DLL_DECL type FLAGS_##name; \
} \ } \
using FLAG__namespace_do_not_use_directly_use_DECLARE_##tn##_instead::FLAGS_##name using FLAG__namespace_do_not_use_directly_use_DECLARE_##tn##_instead::FLAGS_##name
#define DEFINE_VARIABLE(type, name, value, meaning, tn) \ #define DEFINE_VARIABLE(type, name, value, meaning, tn) \
namespace FLAG__namespace_do_not_use_directly_use_DECLARE_##tn##_instead { \ namespace FLAG__namespace_do_not_use_directly_use_DECLARE_##tn##_instead { \
type FLAGS_##name(value); \ GOOGLE_GLOG_DLL_DECL type FLAGS_##name(value); \
char FLAGS_no##name; \ char FLAGS_no##name; \
} \ } \
using FLAG__namespace_do_not_use_directly_use_DECLARE_##tn##_instead::FLAGS_##name using FLAG__namespace_do_not_use_directly_use_DECLARE_##tn##_instead::FLAGS_##name
...@@ -90,12 +90,13 @@ ...@@ -90,12 +90,13 @@
// std::string, which doesn't play nicely with our FLAG__namespace hackery. // std::string, which doesn't play nicely with our FLAG__namespace hackery.
#define DECLARE_string(name) \ #define DECLARE_string(name) \
namespace FLAG__namespace_do_not_use_directly_use_DECLARE_string_instead { \ namespace FLAG__namespace_do_not_use_directly_use_DECLARE_string_instead { \
extern std::string FLAGS_##name; \ extern GOOGLE_GLOG_DLL_DECL std::string FLAGS_##name; \
} \ } \
using FLAG__namespace_do_not_use_directly_use_DECLARE_string_instead::FLAGS_##name using FLAG__namespace_do_not_use_directly_use_DECLARE_string_instead::FLAGS_##name
#define DEFINE_string(name, value, meaning) \ #define DEFINE_string(name, value, meaning) \
namespace FLAG__namespace_do_not_use_directly_use_DECLARE_string_instead { \ namespace FLAG__namespace_do_not_use_directly_use_DECLARE_string_instead { \
std::string FLAGS_##name(EnvToString("GLOG_" #name, value)); \ GOOGLE_GLOG_DLL_DECL std::string \
FLAGS_##name(EnvToString("GLOG_" #name, value)); \
char FLAGS_no##name; \ char FLAGS_no##name; \
} \ } \
using FLAG__namespace_do_not_use_directly_use_DECLARE_string_instead::FLAGS_##name using FLAG__namespace_do_not_use_directly_use_DECLARE_string_instead::FLAGS_##name
......
...@@ -12,6 +12,12 @@ ...@@ -12,6 +12,12 @@
/* Define to 1 if you have the <execinfo.h> header file. */ /* Define to 1 if you have the <execinfo.h> header file. */
#undef HAVE_EXECINFO_H #undef HAVE_EXECINFO_H
/* Define if you have the `fcntl' function */
#undef HAVE_FCNTL
/* Define to 1 if you have the <glob.h> header file. */
#undef HAVE_GLOB_H
/* Define to 1 if you have the <inttypes.h> header file. */ /* Define to 1 if you have the <inttypes.h> header file. */
#undef HAVE_INTTYPES_H #undef HAVE_INTTYPES_H
...@@ -36,6 +42,9 @@ ...@@ -36,6 +42,9 @@
/* Define if you have POSIX threads libraries and header files. */ /* Define if you have POSIX threads libraries and header files. */
#undef HAVE_PTHREAD #undef HAVE_PTHREAD
/* Define to 1 if you have the <pwd.h> header file. */
#undef HAVE_PWD_H
/* define if the compiler implements pthread_rwlock_* */ /* define if the compiler implements pthread_rwlock_* */
#undef HAVE_RWLOCK #undef HAVE_RWLOCK
...@@ -57,15 +66,24 @@ ...@@ -57,15 +66,24 @@
/* Define to 1 if you have the <syscall.h> header file. */ /* Define to 1 if you have the <syscall.h> header file. */
#undef HAVE_SYSCALL_H #undef HAVE_SYSCALL_H
/* Define to 1 if you have the <syslog.h> header file. */
#undef HAVE_SYSLOG_H
/* Define to 1 if you have the <sys/stat.h> header file. */ /* Define to 1 if you have the <sys/stat.h> header file. */
#undef HAVE_SYS_STAT_H #undef HAVE_SYS_STAT_H
/* Define to 1 if you have the <sys/syscall.h> header file. */ /* Define to 1 if you have the <sys/syscall.h> header file. */
#undef HAVE_SYS_SYSCALL_H #undef HAVE_SYS_SYSCALL_H
/* Define to 1 if you have the <sys/time.h> header file. */
#undef HAVE_SYS_TIME_H
/* Define to 1 if you have the <sys/types.h> header file. */ /* Define to 1 if you have the <sys/types.h> header file. */
#undef HAVE_SYS_TYPES_H #undef HAVE_SYS_TYPES_H
/* Define to 1 if you have the <sys/utsname.h> header file. */
#undef HAVE_SYS_UTSNAME_H
/* Define to 1 if you have the <ucontext.h> header file. */ /* Define to 1 if you have the <ucontext.h> header file. */
#undef HAVE_UCONTEXT_H #undef HAVE_UCONTEXT_H
......
// Copyright (c) 2008, Google Inc.
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following disclaimer
// in the documentation and/or other materials provided with the
// distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// ---
// All Rights Reserved.
//
// Author: Craig Silverstein
// Copied from google-perftools and modified by Shinichiro Hamaji
//
// This file is needed for windows -- unittests are not part of the
// glog dll, but still want to include config.h just like the
// dll does, so they can use internal tools and APIs for testing.
//
// The problem is that config.h declares GOOGLE_GLOG_DLL_DECL to be
// for exporting symbols, but the unittest needs to *import* symbols
// (since it's not the dll).
//
// The solution is to have this file, which is just like config.h but
// sets GOOGLE_GLOG_DLL_DECL to do a dllimport instead of a dllexport.
//
// The reason we need this extra GOOGLE_GLOG_DLL_DECL_FOR_UNITTESTS
// variable is in case people want to set GOOGLE_GLOG_DLL_DECL explicitly
// to something other than __declspec(dllexport). In that case, they
// may want to use something other than __declspec(dllimport) for the
// unittest case. For that, we allow folks to define both
// GOOGLE_GLOG_DLL_DECL and GOOGLE_GLOG_DLL_DECL_FOR_UNITTESTS explicitly.
//
// NOTE: This file is equivalent to config.h on non-windows systems,
// which never defined GOOGLE_GLOG_DLL_DECL_FOR_UNITTESTS and always
// define GOOGLE_GLOG_DLL_DECL to the empty string.
#include "config.h"
#undef GOOGLE_GLOG_DLL_DECL
#ifdef GOOGLE_GLOG_DLL_DECL_FOR_UNITTESTS
# define GOOGLE_GLOG_DLL_DECL GOOGLE_GLOG_DLL_DECL_FOR_UNITTESTS
#else
// if DLL_DECL_FOR_UNITTESTS isn't defined, use ""
# define GOOGLE_GLOG_DLL_DECL
#endif
...@@ -3,6 +3,15 @@ ...@@ -3,6 +3,15 @@
#ifndef BASE_LOG_SEVERITY_H__ #ifndef BASE_LOG_SEVERITY_H__
#define BASE_LOG_SEVERITY_H__ #define BASE_LOG_SEVERITY_H__
// Annoying stuff for windows -- makes sure clients can import these functions
#ifndef GOOGLE_GLOG_DLL_DECL
# ifdef _WIN32
# define GOOGLE_GLOG_DLL_DECL __declspec(dllimport)
# else
# define GOOGLE_GLOG_DLL_DECL
# endif
#endif
// Variables of type LogSeverity are widely taken to lie in the range // Variables of type LogSeverity are widely taken to lie in the range
// [0, NUM_SEVERITIES-1]. Be careful to preserve this assumption if // [0, NUM_SEVERITIES-1]. Be careful to preserve this assumption if
// you ever need to change their values or add a new severity. // you ever need to change their values or add a new severity.
...@@ -17,7 +26,7 @@ const int INFO = 0, WARNING = 1, ERROR = 2, FATAL = 3, NUM_SEVERITIES = 4; ...@@ -17,7 +26,7 @@ const int INFO = 0, WARNING = 1, ERROR = 2, FATAL = 3, NUM_SEVERITIES = 4;
#define DFATAL_LEVEL FATAL #define DFATAL_LEVEL FATAL
#endif #endif
extern const char* const LogSeverityNames[NUM_SEVERITIES]; extern GOOGLE_GLOG_DLL_DECL const char* const LogSeverityNames[NUM_SEVERITIES];
// NDEBUG usage helpers related to (RAW_)DCHECK: // NDEBUG usage helpers related to (RAW_)DCHECK:
// //
......
This diff is collapsed.
...@@ -13,6 +13,15 @@ ...@@ -13,6 +13,15 @@
#include "glog/log_severity.h" #include "glog/log_severity.h"
#include "glog/vlog_is_on.h" #include "glog/vlog_is_on.h"
// Annoying stuff for windows -- makes sure clients can import these functions
#ifndef GOOGLE_GLOG_DLL_DECL
# ifdef _WIN32
# define GOOGLE_GLOG_DLL_DECL __declspec(dllimport)
# else
# define GOOGLE_GLOG_DLL_DECL
# endif
#endif
// This is similar to LOG(severity) << format... and VLOG(level) << format.., // This is similar to LOG(severity) << format... and VLOG(level) << format..,
// but // but
// * it is to be used ONLY by low-level modules that can't use normal LOG() // * it is to be used ONLY by low-level modules that can't use normal LOG()
...@@ -72,15 +81,17 @@ ...@@ -72,15 +81,17 @@
// Logs format... at "severity" level, reporting it // Logs format... at "severity" level, reporting it
// as called from file:line. // as called from file:line.
// This does not allocate memory or acquire locks. // This does not allocate memory or acquire locks.
void RawLog__(LogSeverity severity, const char* file, int line, GOOGLE_GLOG_DLL_DECL void RawLog__(LogSeverity severity,
const char* format, ...) const char* file,
int line,
const char* format, ...)
@ac_cv___attribute___printf_4_5@; @ac_cv___attribute___printf_4_5@;
// Hack to propagate time information into this module so that // Hack to propagate time information into this module so that
// this module does not have to directly call localtime_r(), // this module does not have to directly call localtime_r(),
// which could allocate memory. // which could allocate memory.
extern "C" struct ::tm; extern "C" struct ::tm;
void RawLog__SetLastTime(const struct ::tm& t); GOOGLE_GLOG_DLL_DECL void RawLog__SetLastTime(const struct ::tm& t);
@ac_google_end_namespace@ @ac_google_end_namespace@
......
...@@ -35,6 +35,15 @@ ...@@ -35,6 +35,15 @@
#include "glog/log_severity.h" #include "glog/log_severity.h"
// Annoying stuff for windows -- makes sure clients can import these functions
#ifndef GOOGLE_GLOG_DLL_DECL
# ifdef _WIN32
# define GOOGLE_GLOG_DLL_DECL __declspec(dllimport)
# else
# define GOOGLE_GLOG_DLL_DECL
# endif
#endif
#if defined(__GNUC__) #if defined(__GNUC__)
// We emit an anonymous static int* variable at every VLOG_IS_ON(n) site. // We emit an anonymous static int* variable at every VLOG_IS_ON(n) site.
// (Normally) the first time every VLOG_IS_ON(n) site is hit, // (Normally) the first time every VLOG_IS_ON(n) site is hit,
...@@ -63,7 +72,8 @@ ...@@ -63,7 +72,8 @@
// one needs to supply the exact --vmodule pattern that applied to them. // one needs to supply the exact --vmodule pattern that applied to them.
// (If no --vmodule pattern applied to them // (If no --vmodule pattern applied to them
// the value of FLAGS_v will continue to control them.) // the value of FLAGS_v will continue to control them.)
extern int SetVLOGLevel(const char* module_pattern, int log_level); extern GOOGLE_GLOG_DLL_DECL int SetVLOGLevel(const char* module_pattern,
int log_level);
// Various declarations needed for VLOG_IS_ON above: ========================= // Various declarations needed for VLOG_IS_ON above: =========================
...@@ -81,9 +91,10 @@ extern @ac_google_namespace@::int32 kLogSiteUninitialized; ...@@ -81,9 +91,10 @@ extern @ac_google_namespace@::int32 kLogSiteUninitialized;
// verbose_level is the argument to VLOG_IS_ON // verbose_level is the argument to VLOG_IS_ON
// We will return the return value for VLOG_IS_ON // We will return the return value for VLOG_IS_ON
// and if possible set *site_flag appropriately. // and if possible set *site_flag appropriately.
extern bool InitVLOG3__(@ac_google_namespace@::int32** site_flag, extern GOOGLE_GLOG_DLL_DECL bool InitVLOG3__(
@ac_google_namespace@::int32* site_default, @ac_google_namespace@::int32** site_flag,
const char* fname, @ac_google_namespace@::int32* site_default,
@ac_google_namespace@::int32 verbose_level); const char* fname,
@ac_google_namespace@::int32 verbose_level);
#endif // BASE_VLOG_IS_ON_H_ #endif // BASE_VLOG_IS_ON_H_
...@@ -15,7 +15,9 @@ ...@@ -15,7 +15,9 @@
#include <sys/types.h> #include <sys/types.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <fcntl.h> #include <fcntl.h>
#include <unistd.h> #ifdef HAVE_UNISTD_H
# include <unistd.h>
#endif
#include "base/commandlineflags.h" #include "base/commandlineflags.h"
...@@ -23,15 +25,44 @@ using std::map; ...@@ -23,15 +25,44 @@ using std::map;
using std::string; using std::string;
using std::vector; using std::vector;
DEFINE_string(test_tmpdir, "/tmp", "Dir we use for temp files"); _START_GOOGLE_NAMESPACE_
DEFINE_string(test_srcdir, ".",
extern GOOGLE_GLOG_DLL_DECL void (*g_logging_fail_func)();
_END_GOOGLE_NAMESPACE_
#undef GOOGLE_GLOG_DLL_DECL
#define GOOGLE_GLOG_DLL_DECL
static string GetTempDir() {
#ifndef OS_WINDOWS
return "/tmp";
#else
char tmp[MAX_PATH];
GetTempPathA(MAX_PATH, tmp);
return tmp;
#endif
}
#ifdef OS_WINDOWS
// The test will run in glog/vsproject/<project name>
// (e.g., glog/vsproject/logging_unittest).
static const char TEST_SRC_DIR[] = "../..";
#else
static const char TEST_SRC_DIR[] = ".";
#endif
DEFINE_string(test_tmpdir, GetTempDir(), "Dir we use for temp files");
DEFINE_string(test_srcdir, TEST_SRC_DIR,
"Source-dir root, needed to find glog_unittest_flagfile"); "Source-dir root, needed to find glog_unittest_flagfile");
#ifdef NDEBUG
DEFINE_int32(benchmark_iters, 100000000, "Number of iterations per benchmark"); DEFINE_int32(benchmark_iters, 100000000, "Number of iterations per benchmark");
#else
DEFINE_int32(benchmark_iters, 1000000, "Number of iterations per benchmark");
#endif
_START_GOOGLE_NAMESPACE_ _START_GOOGLE_NAMESPACE_
extern void (*g_logging_fail_func)();
// The following is some bare-bones testing infrastructure // The following is some bare-bones testing infrastructure
#define EXPECT_TRUE(cond) \ #define EXPECT_TRUE(cond) \
...@@ -96,6 +127,10 @@ static void CalledAbort() { ...@@ -96,6 +127,10 @@ static void CalledAbort() {
longjmp(g_jmp_buf, 1); longjmp(g_jmp_buf, 1);
} }
#ifdef OS_WINDOWS
// TODO(hamaji): Death test somehow doesn't work in Windows.
#define ASSERT_DEATH(fn, msg)
#else
#define ASSERT_DEATH(fn, msg) \ #define ASSERT_DEATH(fn, msg) \
do { \ do { \
g_called_abort = false; \ g_called_abort = false; \
...@@ -110,6 +145,7 @@ static void CalledAbort() { ...@@ -110,6 +145,7 @@ static void CalledAbort() {
exit(1); \ exit(1); \
} \ } \
} while (0) } while (0)
#endif
#ifdef NDEBUG #ifdef NDEBUG
#define ASSERT_DEBUG_DEATH(fn, msg) #define ASSERT_DEBUG_DEATH(fn, msg)
...@@ -307,11 +343,10 @@ static string MungeLine(const string& line) { ...@@ -307,11 +343,10 @@ static string MungeLine(const string& line) {
iss >> logcode_date; iss >> logcode_date;
while (!IsLoggingPrefix(logcode_date)) { while (!IsLoggingPrefix(logcode_date)) {
before += " " + logcode_date; before += " " + logcode_date;
if (iss.eof()) { if (!(iss >> logcode_date)) {
// We cannot find the header of log output. // We cannot find the header of log output.
return before; return before;
} }
iss >> logcode_date;
} }
if (!before.empty()) before += " "; if (!before.empty()) before += " ";
iss >> time; iss >> time;
...@@ -354,6 +389,8 @@ static string Munge(const string& filename) { ...@@ -354,6 +389,8 @@ static string Munge(const string& filename) {
char null_str[256]; char null_str[256];
sprintf(null_str, "%p", NULL); sprintf(null_str, "%p", NULL);
StringReplace(&line, "__NULLP__", null_str); StringReplace(&line, "__NULLP__", null_str);
// Remove 0x prefix produced by %p. VC++ doesn't put the prefix.
StringReplace(&line, " 0x", " ");
char errmsg_buf[100]; char errmsg_buf[100];
posix_strerror_r(0, errmsg_buf, sizeof(errmsg_buf)); posix_strerror_r(0, errmsg_buf, sizeof(errmsg_buf));
...@@ -429,16 +466,32 @@ struct FlagSaver { ...@@ -429,16 +466,32 @@ struct FlagSaver {
}; };
#endif #endif
// TODO(hamaji): Make it portable.
class Thread { class Thread {
public: public:
void SetJoinable(bool joinable) {} void SetJoinable(bool joinable) {}
#if defined(HAVE_PTHREAD)
void Start() { void Start() {
pthread_create(&th_, NULL, &Thread::InvokeThread, this); pthread_create(&th_, NULL, &Thread::InvokeThread, this);
} }
void Join() { void Join() {
pthread_join(th_, NULL); pthread_join(th_, NULL);
} }
#elif defined(OS_WINDOWS)
void Start() {
handle_ = CreateThread(NULL,
0,
(LPTHREAD_START_ROUTINE)&Thread::InvokeThread,
(LPVOID)this,
0,
&th_);
CHECK(handle_) << "CreateThread";
}
void Join() {
WaitForSingleObject(handle_, INFINITE);
}
#else
# error No thread implementation.
#endif
protected: protected:
virtual void Run() = 0; virtual void Run() = 0;
...@@ -450,11 +503,17 @@ class Thread { ...@@ -450,11 +503,17 @@ class Thread {
} }
pthread_t th_; pthread_t th_;
#ifdef OS_WINDOWS
HANDLE handle_;
#endif
}; };
// TODO(hamaji): Make it portable.
static void SleepForMilliseconds(int t) { static void SleepForMilliseconds(int t) {
#ifndef OS_WINDOWS
usleep(t * 1000); usleep(t * 1000);
#else
Sleep(t);
#endif
} }
// Add hook for operator new to ensure there are no memory allocation. // Add hook for operator new to ensure there are no memory allocation.
......
This diff is collapsed.
...@@ -4,12 +4,17 @@ ...@@ -4,12 +4,17 @@
// //
// Author: Ray Sidney // Author: Ray Sidney
#include "config_for_unittests.h"
#include "utilities.h" #include "utilities.h"
#include <fcntl.h> #include <fcntl.h>
#include <glob.h> #ifdef HAVE_GLOB_H
# include <glob.h>
#endif
#include <sys/stat.h> #include <sys/stat.h>
#include <unistd.h> #ifdef HAVE_UNISTD_H
# include <unistd.h>
#endif
#include <iomanip> #include <iomanip>
#include <iostream> #include <iostream>
...@@ -356,7 +361,7 @@ void TestLogString() { ...@@ -356,7 +361,7 @@ void TestLogString() {
LOG_STRING(WARNING, no_errors) << "LOG_STRING: " << "reported warning"; LOG_STRING(WARNING, no_errors) << "LOG_STRING: " << "reported warning";
LOG_STRING(ERROR, NULL) << "LOG_STRING: " << "reported error"; LOG_STRING(ERROR, NULL) << "LOG_STRING: " << "reported error";
for (int i = 0; i < errors.size(); ++i) { for (size_t i = 0; i < errors.size(); ++i) {
LOG(INFO) << "Captured by LOG_STRING: " << errors[i]; LOG(INFO) << "Captured by LOG_STRING: " << errors[i];
} }
} }
...@@ -386,7 +391,7 @@ void TestLogSink() { ...@@ -386,7 +391,7 @@ void TestLogSink() {
LOG_TO_SINK(NULL, ERROR) << "LOG_TO_SINK: " << "reported error"; LOG_TO_SINK(NULL, ERROR) << "LOG_TO_SINK: " << "reported error";
LOG(INFO) << "Captured by LOG_TO_SINK:"; LOG(INFO) << "Captured by LOG_TO_SINK:";
for (int i = 0; i < sink.errors.size(); ++i) { for (size_t i = 0; i < sink.errors.size(); ++i) {
LogMessage("foo", LogMessage::kNoLogPrefix, INFO).stream() LogMessage("foo", LogMessage::kNoLogPrefix, INFO).stream()
<< sink.errors[i]; << sink.errors[i];
} }
...@@ -481,6 +486,7 @@ TEST(DeathCheckNN, Simple) { ...@@ -481,6 +486,7 @@ TEST(DeathCheckNN, Simple) {
// Get list of file names that match pattern // Get list of file names that match pattern
static void GetFiles(const string& pattern, vector<string>* files) { static void GetFiles(const string& pattern, vector<string>* files) {
files->clear(); files->clear();
#if defined(HAVE_GLOB_H)
glob_t g; glob_t g;
const int r = glob(pattern.c_str(), 0, NULL, &g); const int r = glob(pattern.c_str(), 0, NULL, &g);
CHECK((r == 0) || (r == GLOB_NOMATCH)) << ": error matching " << pattern; CHECK((r == 0) || (r == GLOB_NOMATCH)) << ": error matching " << pattern;
...@@ -488,13 +494,32 @@ static void GetFiles(const string& pattern, vector<string>* files) { ...@@ -488,13 +494,32 @@ static void GetFiles(const string& pattern, vector<string>* files) {
files->push_back(string(g.gl_pathv[i])); files->push_back(string(g.gl_pathv[i]));
} }
globfree(&g); globfree(&g);
#elif defined(OS_WINDOWS)
WIN32_FIND_DATAA data;
HANDLE handle = FindFirstFileA(pattern.c_str(), &data);
size_t index = pattern.rfind('\\');
if (index == string::npos) {
LOG(FATAL) << "No directory separator.";
}
const string dirname = pattern.substr(0, index + 1);
if (FAILED(handle)) {
// Finding no files is OK.
return;
}
do {
files->push_back(dirname + data.cFileName);
} while (FindNextFileA(handle, &data));
LOG_SYSRESULT(FindClose(handle));
#else
# error There is no way to do glob.
#endif
} }
// Delete files patching pattern // Delete files patching pattern
static void DeleteFiles(const string& pattern) { static void DeleteFiles(const string& pattern) {
vector<string> files; vector<string> files;
GetFiles(pattern, &files); GetFiles(pattern, &files);
for (int i = 0; i < files.size(); i++) { for (size_t i = 0; i < files.size(); i++) {
CHECK(unlink(files[i].c_str()) == 0) << ": " << strerror(errno); CHECK(unlink(files[i].c_str()) == 0) << ": " << strerror(errno);
} }
} }
...@@ -519,18 +544,22 @@ static void CheckFile(const string& name, const string& expected_string) { ...@@ -519,18 +544,22 @@ static void CheckFile(const string& name, const string& expected_string) {
static void TestBasename() { static void TestBasename() {
fprintf(stderr, "==== Test setting log file basename\n"); fprintf(stderr, "==== Test setting log file basename\n");
string dest = FLAGS_test_tmpdir + "/logging_test_basename"; const string dest = FLAGS_test_tmpdir + "/logging_test_basename";
DeleteFiles(dest + "*"); DeleteFiles(dest + "*");
SetLogDestination(INFO, dest.c_str()); SetLogDestination(INFO, dest.c_str());
LOG(INFO) << "message to new base"; LOG(INFO) << "message to new base";
FlushLogFiles(INFO); FlushLogFiles(INFO);
CheckFile(dest, "message to new base"); CheckFile(dest, "message to new base");
// Release file handle for the destination file to unlock the file in Windows.
LogToStderr();
DeleteFiles(dest + "*"); DeleteFiles(dest + "*");
} }
static void TestSymlink() { static void TestSymlink() {
#ifndef OS_WINDOWS
fprintf(stderr, "==== Test setting log file symlink\n"); fprintf(stderr, "==== Test setting log file symlink\n");
string dest = FLAGS_test_tmpdir + "/logging_test_symlink"; string dest = FLAGS_test_tmpdir + "/logging_test_symlink";
string sym = FLAGS_test_tmpdir + "/symlinkbase"; string sym = FLAGS_test_tmpdir + "/symlinkbase";
...@@ -545,6 +574,7 @@ static void TestSymlink() { ...@@ -545,6 +574,7 @@ static void TestSymlink() {
DeleteFiles(dest + "*"); DeleteFiles(dest + "*");
DeleteFiles(sym + "*"); DeleteFiles(sym + "*");
#endif
} }
static void TestExtension() { static void TestExtension() {
...@@ -564,6 +594,8 @@ static void TestExtension() { ...@@ -564,6 +594,8 @@ static void TestExtension() {
CHECK_EQ(filenames.size(), 1); CHECK_EQ(filenames.size(), 1);
CHECK(strstr(filenames[0].c_str(), "specialextension") != NULL); CHECK(strstr(filenames[0].c_str(), "specialextension") != NULL);
// Release file handle for the destination file to unlock the file in Windows.
LogToStderr();
DeleteFiles(dest + "*"); DeleteFiles(dest + "*");
} }
...@@ -633,9 +665,10 @@ static void TestOneTruncate(const char *path, int64 limit, int64 keep, ...@@ -633,9 +665,10 @@ static void TestOneTruncate(const char *path, int64 limit, int64 keep,
CHECK_ERR(lseek(fd, 0, SEEK_SET)); CHECK_ERR(lseek(fd, 0, SEEK_SET));
// File should contain the suffix of the original file // File should contain the suffix of the original file
char buf[statbuf.st_size + 1]; int buf_size = statbuf.st_size + 1;
char* buf = new char[buf_size];
memset(buf, 0, sizeof(buf)); memset(buf, 0, sizeof(buf));
CHECK_ERR(read(fd, buf, sizeof(buf))); CHECK_ERR(read(fd, buf, buf_size));
const char *p = buf; const char *p = buf;
int64 checked = 0; int64 checked = 0;
...@@ -645,9 +678,11 @@ static void TestOneTruncate(const char *path, int64 limit, int64 keep, ...@@ -645,9 +678,11 @@ static void TestOneTruncate(const char *path, int64 limit, int64 keep,
checked += bytes; checked += bytes;
} }
close(fd); close(fd);
delete[] buf;
} }
static void TestTruncate() { static void TestTruncate() {
#ifdef HAVE_UNISTD_H
fprintf(stderr, "==== Test log truncation\n"); fprintf(stderr, "==== Test log truncation\n");
string path = FLAGS_test_tmpdir + "/truncatefile"; string path = FLAGS_test_tmpdir + "/truncatefile";
...@@ -663,8 +698,10 @@ static void TestTruncate() { ...@@ -663,8 +698,10 @@ static void TestTruncate() {
TestOneTruncate(path.c_str(), 10, 50, 0, 10, 10); TestOneTruncate(path.c_str(), 10, 50, 0, 10, 10);
TestOneTruncate(path.c_str(), 50, 100, 0, 30, 30); TestOneTruncate(path.c_str(), 50, 100, 0, 30, 30);
// MacOSX 10.4 doesn't fail in this case. Let's just ignore this test. // MacOSX 10.4 doesn't fail in this case.
#if !defined(OS_MACOSX) // Windows doesn't have symlink.
// Let's just ignore this test for these cases.
#if !defined(OS_MACOSX) && !defined(OS_WINDOWS)
// Through a symlink should fail to truncate // Through a symlink should fail to truncate
string linkname = path + ".link"; string linkname = path + ".link";
unlink(linkname.c_str()); unlink(linkname.c_str());
...@@ -681,12 +718,17 @@ static void TestTruncate() { ...@@ -681,12 +718,17 @@ static void TestTruncate() {
snprintf(fdpath, sizeof(fdpath), "/proc/self/fd/%d", fd); snprintf(fdpath, sizeof(fdpath), "/proc/self/fd/%d", fd);
TestOneTruncate(fdpath, 10, 10, 10, 10, 10); TestOneTruncate(fdpath, 10, 10, 10, 10, 10);
#endif #endif
#endif
} }
_START_GOOGLE_NAMESPACE_ _START_GOOGLE_NAMESPACE_
namespace glog_internal_namespace_ {
extern // in logging.cc extern // in logging.cc
bool SafeFNMatch_(const char* pattern, size_t patt_len, bool SafeFNMatch_(const char* pattern, size_t patt_len,
const char* str, size_t str_len); const char* str, size_t str_len);
} // namespace glog_internal_namespace_
using glog_internal_namespace_::SafeFNMatch_;
_END_GOOGLE_NAMESPACE_ _END_GOOGLE_NAMESPACE_
static bool WrapSafeFNMatch(string pattern, string str) { static bool WrapSafeFNMatch(string pattern, string str) {
...@@ -865,7 +907,7 @@ static void TestLogSinkWaitTillSent() { ...@@ -865,7 +907,7 @@ static void TestLogSinkWaitTillSent() {
LOG(WARNING) << "Message 3"; LOG(WARNING) << "Message 3";
SleepForMilliseconds(60); SleepForMilliseconds(60);
} }
for (int i = 0; i < global_messages.size(); ++i) { for (size_t i = 0; i < global_messages.size(); ++i) {
LOG(INFO) << "Sink capture: " << global_messages[i]; LOG(INFO) << "Sink capture: " << global_messages[i];
} }
CHECK_EQ(global_messages.size(), 3); CHECK_EQ(global_messages.size(), 3);
...@@ -874,12 +916,13 @@ static void TestLogSinkWaitTillSent() { ...@@ -874,12 +916,13 @@ static void TestLogSinkWaitTillSent() {
TEST(Strerror, logging) { TEST(Strerror, logging) {
int errcode = EINTR; int errcode = EINTR;
char *msg = strdup(strerror(errcode)); char *msg = strdup(strerror(errcode));
char buf[strlen(msg) + 1]; int buf_size = strlen(msg) + 1;
char *buf = new char[buf_size];
CHECK_EQ(posix_strerror_r(errcode, NULL, 0), -1); CHECK_EQ(posix_strerror_r(errcode, NULL, 0), -1);
buf[0] = 'A'; buf[0] = 'A';
CHECK_EQ(posix_strerror_r(errcode, buf, 0), -1); CHECK_EQ(posix_strerror_r(errcode, buf, 0), -1);
CHECK_EQ(buf[0], 'A'); CHECK_EQ(buf[0], 'A');
CHECK_EQ(posix_strerror_r(errcode, NULL, sizeof(buf)), -1); CHECK_EQ(posix_strerror_r(errcode, NULL, buf_size), -1);
#if defined(OS_MACOSX) || defined(OS_FREEBSD) #if defined(OS_MACOSX) || defined(OS_FREEBSD)
// MacOSX or FreeBSD considers this case is an error since there is // MacOSX or FreeBSD considers this case is an error since there is
// no enough space. // no enough space.
...@@ -888,9 +931,10 @@ TEST(Strerror, logging) { ...@@ -888,9 +931,10 @@ TEST(Strerror, logging) {
CHECK_EQ(posix_strerror_r(errcode, buf, 1), 0); CHECK_EQ(posix_strerror_r(errcode, buf, 1), 0);
#endif #endif
CHECK_STREQ(buf, ""); CHECK_STREQ(buf, "");
CHECK_EQ(posix_strerror_r(errcode, buf, sizeof(buf)), 0); CHECK_EQ(posix_strerror_r(errcode, buf, buf_size), 0);
CHECK_STREQ(buf, msg); CHECK_STREQ(buf, msg);
free(msg); free(msg);
delete[] buf;
} }
// Simple routines to look at the sizes of generated code for LOG(FATAL) and // Simple routines to look at the sizes of generated code for LOG(FATAL) and
......
...@@ -17,15 +17,9 @@ ...@@ -17,15 +17,9 @@
#elif defined(HAVE_SYS_SYSCALL_H) #elif defined(HAVE_SYS_SYSCALL_H)
#include <sys/syscall.h> // for syscall() #include <sys/syscall.h> // for syscall()
#endif #endif
#include <unistd.h> #ifdef HAVE_UNISTD_H
# include <unistd.h>
#if defined(OS_MACOSX)
#ifndef __DARWIN_UNIX03
#define __DARWIN_UNIX03 // tells libgen.h to define basename()
#endif #endif
#endif // OS_MACOSX
#include <libgen.h> // basename()
_START_GOOGLE_NAMESPACE_ _START_GOOGLE_NAMESPACE_
...@@ -82,13 +76,13 @@ void RawLog__(LogSeverity severity, const char* file, int line, ...@@ -82,13 +76,13 @@ void RawLog__(LogSeverity severity, const char* file, int line,
DoRawLog(&buf, &size, "%c%02d%02d %02d%02d%02d %s:%d] RAW: ", DoRawLog(&buf, &size, "%c%02d%02d %02d%02d%02d %s:%d] RAW: ",
LogSeverityNames[severity][0], LogSeverityNames[severity][0],
1 + t.tm_mon, t.tm_mday, t.tm_hour, t.tm_min, t.tm_sec, 1 + t.tm_mon, t.tm_mday, t.tm_hour, t.tm_min, t.tm_sec,
basename(const_cast<char *>(file)), line); const_basename(const_cast<char *>(file)), line);
} else { } else {
DoRawLog(&buf, &size, "%c%02d%02d %02d%02d%02d %08x %s:%d] RAW: ", DoRawLog(&buf, &size, "%c%02d%02d %02d%02d%02d %08x %s:%d] RAW: ",
LogSeverityNames[severity][0], LogSeverityNames[severity][0],
1 + t.tm_mon, t.tm_mday, t.tm_hour, t.tm_min, t.tm_sec, 1 + t.tm_mon, t.tm_mday, t.tm_hour, t.tm_min, t.tm_sec,
int(pthread_self()), int(pthread_self()),
basename(const_cast<char *>(file)), line); const_basename(const_cast<char *>(file)), line);
} }
va_list ap; va_list ap;
va_start(ap, format); va_start(ap, format);
......
...@@ -4,7 +4,9 @@ ...@@ -4,7 +4,9 @@
#include "utilities.h" #include "utilities.h"
#include <signal.h> #include <signal.h>
#include <sys/time.h> #ifdef HAVE_SYS_TIME_H
# include <sys/time.h>
#endif
#include <time.h> #include <time.h>
#include "base/googleinit.h" #include "base/googleinit.h"
...@@ -138,9 +140,17 @@ bool is_default_thread() { ...@@ -138,9 +140,17 @@ bool is_default_thread() {
int64 CycleClock_Now() { int64 CycleClock_Now() {
// TODO(hamaji): temporary impementation - it might be too slow. // TODO(hamaji): temporary impementation - it might be too slow.
#ifdef OS_WINDOWS
SYSTEMTIME now;
GetSystemTime(&now);
return (static_cast<int64>(now.wSecond) * 1000000 +
static_cast<int64>(now.wMilliseconds) * 1000);
#else
// TODO(hamaji): temporary impementation - it might be too slow.
struct timeval tv; struct timeval tv;
gettimeofday(&tv, NULL); gettimeofday(&tv, NULL);
return static_cast<int64>(tv.tv_sec) * 1000000 + tv.tv_usec; return static_cast<int64>(tv.tv_sec) * 1000000 + tv.tv_usec;
#endif
} }
int64 UsecToCycles(int64 usec) { int64 UsecToCycles(int64 usec) {
...@@ -152,6 +162,15 @@ int32 GetMainThreadPid() { ...@@ -152,6 +162,15 @@ int32 GetMainThreadPid() {
return g_main_thread_pid; return g_main_thread_pid;
} }
const char* const_basename(const char* filepath) {
const char* base = strrchr(filepath, '/');
#ifdef OS_WINDOWS // Look for either path separator in Windows
if (!base)
base = strrchr(filepath, '\\');
#endif
return base ? (base+1) : filepath;
}
static string g_my_user_name; static string g_my_user_name;
const string& MyUserName() { const string& MyUserName() {
return g_my_user_name; return g_my_user_name;
......
...@@ -40,6 +40,10 @@ ...@@ -40,6 +40,10 @@
#include <string> #include <string>
#ifdef OS_WINDOWS
# include "port.h"
#endif
#include "config.h" #include "config.h"
#include "glog/logging.h" #include "glog/logging.h"
...@@ -116,6 +120,10 @@ int32 GetMainThreadPid(); ...@@ -116,6 +120,10 @@ int32 GetMainThreadPid();
const std::string& MyUserName(); const std::string& MyUserName();
// Get the part of filepath after the last path separator.
// (Doesn't modify filepath, contrary to basename() in libgen.h.)
const char* const_basename(const char* filepath);
// Wrapper of __sync_val_compare_and_swap. If the GCC extension isn't // Wrapper of __sync_val_compare_and_swap. If the GCC extension isn't
// defined, we try the CPU specific logics (we only support x86 and // defined, we try the CPU specific logics (we only support x86 and
// x86_64 for now) first, then use a naive implementation, which has a // x86_64 for now) first, then use a naive implementation, which has a
......
...@@ -8,7 +8,6 @@ ...@@ -8,7 +8,6 @@
#include <string.h> #include <string.h>
#include <stdlib.h> #include <stdlib.h>
#include <errno.h> #include <errno.h>
#include <fnmatch.h> // fnmatch() is used in obsolete _InitVLOG
#include <cstdio> #include <cstdio>
#include <string> #include <string>
#include "base/commandlineflags.h" #include "base/commandlineflags.h"
...@@ -32,12 +31,16 @@ DEFINE_string(vmodule, "", "per-module verbose level." ...@@ -32,12 +31,16 @@ DEFINE_string(vmodule, "", "per-module verbose level."
_START_GOOGLE_NAMESPACE_ _START_GOOGLE_NAMESPACE_
namespace glog_internal_namespace_ {
// Implementation of fnmatch that does not need 0-termination // Implementation of fnmatch that does not need 0-termination
// of arguments and does not allocate any memory, // of arguments and does not allocate any memory,
// but we only support "*" and "?" wildcards, not the "[...]" patterns. // but we only support "*" and "?" wildcards, not the "[...]" patterns.
// It's not a static function for the unittest. // It's not a static function for the unittest.
bool SafeFNMatch_(const char* pattern, size_t patt_len, GOOGLE_GLOG_DLL_DECL bool SafeFNMatch_(const char* pattern,
const char* str, size_t str_len) { size_t patt_len,
const char* str,
size_t str_len) {
int p = 0; int p = 0;
int s = 0; int s = 0;
while (1) { while (1) {
...@@ -63,6 +66,10 @@ bool SafeFNMatch_(const char* pattern, size_t patt_len, ...@@ -63,6 +66,10 @@ bool SafeFNMatch_(const char* pattern, size_t patt_len,
} }
} }
} // namespace glog_internal_namespace_
using glog_internal_namespace_::SafeFNMatch_;
int32 kLogSiteUninitialized = 1000; int32 kLogSiteUninitialized = 1000;
// List of per-module log levels from FLAGS_vmodule. // List of per-module log levels from FLAGS_vmodule.
......
/* src/config.h.in. Generated from configure.ac by autoheader. */
/* Namespace for Google classes */
#define GOOGLE_NAMESPACE google
/* Define if you have the `dladdr' function */
#undef HAVE_DLADDR
/* Define to 1 if you have the <dlfcn.h> header file. */
#undef HAVE_DLFCN_H
/* Define to 1 if you have the <execinfo.h> header file. */
#undef HAVE_EXECINFO_H
/* Define to 1 if you have the <inttypes.h> header file. */
#undef HAVE_INTTYPES_H
/* Define to 1 if you have the <libunwind.h> header file. */
#undef HAVE_LIBUNWIND_H
/* define if you have google gflags library */
#undef HAVE_LIB_GFLAGS
/* define if you have libunwind */
#undef HAVE_LIB_UNWIND
/* Define to 1 if you have the <memory.h> header file. */
#undef HAVE_MEMORY_H
/* define if the compiler implements namespaces */
#undef HAVE_NAMESPACES
/* Define if you have POSIX threads libraries and header files. */
#undef HAVE_PTHREAD
/* define if the compiler implements pthread_rwlock_* */
#undef HAVE_RWLOCK
/* Define if you have the `sigaltstack' function */
#undef HAVE_SIGALTSTACK
/* Define to 1 if you have the <stdint.h> header file. */
#undef HAVE_STDINT_H
/* Define to 1 if you have the <stdlib.h> header file. */
#undef HAVE_STDLIB_H
/* Define to 1 if you have the <strings.h> header file. */
#undef HAVE_STRINGS_H
/* Define to 1 if you have the <string.h> header file. */
#undef HAVE_STRING_H
/* Define to 1 if you have the <syscall.h> header file. */
#undef HAVE_SYSCALL_H
/* Define to 1 if you have the <sys/stat.h> header file. */
#undef HAVE_SYS_STAT_H
/* Define to 1 if you have the <sys/syscall.h> header file. */
#undef HAVE_SYS_SYSCALL_H
/* Define to 1 if you have the <sys/types.h> header file. */
#undef HAVE_SYS_TYPES_H
/* Define to 1 if you have the <ucontext.h> header file. */
#undef HAVE_UCONTEXT_H
/* Define to 1 if you have the <unistd.h> header file. */
#undef HAVE_UNISTD_H
/* define if the compiler supports using expression for operator */
#undef HAVE_USING_OPERATOR
/* define if your compiler has __attribute__ */
#undef HAVE___ATTRIBUTE__
/* define if your compiler has __builtin_expect */
#undef HAVE___BUILTIN_EXPECT
/* define if your compiler has __sync_val_compare_and_swap */
#undef HAVE___SYNC_VAL_COMPARE_AND_SWAP
/* Name of package */
#undef PACKAGE
/* Define to the address where bug reports for this package should be sent. */
#undef PACKAGE_BUGREPORT
/* Define to the full name of this package. */
#undef PACKAGE_NAME
/* Define to the full name and version of this package. */
#undef PACKAGE_STRING
/* Define to the one symbol short name of this package. */
#undef PACKAGE_TARNAME
/* Define to the version of this package. */
#undef PACKAGE_VERSION
/* How to access the PC from a struct ucontext */
#undef PC_FROM_UCONTEXT
/* Define to necessary symbol if this constant uses a non-standard name on
your system. */
#undef PTHREAD_CREATE_JOINABLE
/* The size of `void *', as computed by sizeof. */
#undef SIZEOF_VOID_P
/* Define to 1 if you have the ANSI C header files. */
#undef STDC_HEADERS
/* the namespace where STL code like vector<> is defined */
#undef STL_NAMESPACE
/* Version number of package */
#undef VERSION
/* Stops putting the code inside the Google namespace */
#define _END_GOOGLE_NAMESPACE_ }
/* Puts following code inside the Google namespace */
#define _START_GOOGLE_NAMESPACE_ namespace google {
/* Always the empty-string on non-windows systems. On windows, should be
"__declspec(dllexport)". This way, when we compile the dll, we export our
functions/classes. It's safe to define this here because config.h is only
used internally, to compile the DLL, and every DLL source file #includes
"config.h" before anything else. */
#ifndef GOOGLE_GLOG_DLL_DECL
# define GOOGLE_GLOG_IS_A_DLL 1 /* not set if you're statically linking */
# define GOOGLE_GLOG_DLL_DECL __declspec(dllexport)
# define GOOGLE_GLOG_DLL_DECL_FOR_UNITTESTS __declspec(dllimport)
#endif
// This file is automatically generated from src/glog/log_severity.h
// using src/windows/preprocess.sh.
// DO NOT EDIT!
// Copyright 2007 Google Inc. All Rights Reserved.
#ifndef BASE_LOG_SEVERITY_H__
#define BASE_LOG_SEVERITY_H__
// Annoying stuff for windows -- makes sure clients can import these functions
#ifndef GOOGLE_GLOG_DLL_DECL
# ifdef _WIN32
# define GOOGLE_GLOG_DLL_DECL __declspec(dllimport)
# else
# define GOOGLE_GLOG_DLL_DECL
# endif
#endif
// Variables of type LogSeverity are widely taken to lie in the range
// [0, NUM_SEVERITIES-1]. Be careful to preserve this assumption if
// you ever need to change their values or add a new severity.
typedef int LogSeverity;
const int INFO = 0, WARNING = 1, ERROR = 2, FATAL = 3, NUM_SEVERITIES = 4;
// DFATAL is FATAL in debug mode, ERROR in normal mode
#ifdef NDEBUG
#define DFATAL_LEVEL ERROR
#else
#define DFATAL_LEVEL FATAL
#endif
extern GOOGLE_GLOG_DLL_DECL const char* const LogSeverityNames[NUM_SEVERITIES];
// NDEBUG usage helpers related to (RAW_)DCHECK:
//
// DEBUG_MODE is for small !NDEBUG uses like
// if (DEBUG_MODE) foo.CheckThatFoo();
// instead of substantially more verbose
// #ifndef NDEBUG
// foo.CheckThatFoo();
// #endif
//
// IF_DEBUG_MODE is for small !NDEBUG uses like
// IF_DEBUG_MODE( string error; )
// DCHECK(Foo(&error)) << error;
// instead of substantially more verbose
// #ifndef NDEBUG
// string error;
// DCHECK(Foo(&error)) << error;
// #endif
//
#ifdef NDEBUG
enum { DEBUG_MODE = 0 };
#define IF_DEBUG_MODE(x)
#else
enum { DEBUG_MODE = 1 };
#define IF_DEBUG_MODE(x) x
#endif
#endif // BASE_LOG_SEVERITY_H__
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
/* Copyright (c) 2008, Google Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* ---
* Author: Craig Silverstein
* Copied from google-perftools and modified by Shinichiro Hamaji
*/
#ifndef _WIN32
# error You should only be including windows/port.cc in a windows environment!
#endif
#include "config.h"
#include <stdarg.h> // for va_list, va_start, va_end
#include <string.h> // for strstr()
#include <assert.h>
#include <string>
#include <vector>
#include "port.h"
using std::string;
using std::vector;
// These call the windows _vsnprintf, but always NUL-terminate.
int safe_vsnprintf(char *str, size_t size, const char *format, va_list ap) {
if (size == 0) // not even room for a \0?
return -1; // not what C99 says to do, but what windows does
str[size-1] = '\0';
return _vsnprintf(str, size-1, format, ap);
}
int snprintf(char *str, size_t size, const char *format, ...) {
va_list ap;
va_start(ap, format);
const int r = vsnprintf(str, size, format, ap);
va_end(ap);
return r;
}
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment