summaryrefslogtreecommitdiffstats
path: root/Source
diff options
context:
space:
mode:
authorBrad King <brad.king@kitware.com>2009-02-24 15:40:18 (GMT)
committerBrad King <brad.king@kitware.com>2009-02-24 15:40:18 (GMT)
commit1936499250425a1d50d70759758c7385b4bdf6d4 (patch)
treeba467f3fdf79068f715b64eb6a72536d9cee2996 /Source
parent6bdc2b5d99313ab6418b07ff91320135907ac261 (diff)
downloadCMake-1936499250425a1d50d70759758c7385b4bdf6d4.zip
CMake-1936499250425a1d50d70759758c7385b4bdf6d4.tar.gz
CMake-1936499250425a1d50d70759758c7385b4bdf6d4.tar.bz2
ENH: Create cmProcessTools to parse child output
This class provides a RunProcess method to run a child process and send its output to an abstract parsing interface. This also provides a simple line parser and logger implementing the parsing interface.
Diffstat (limited to 'Source')
-rw-r--r--Source/CMakeLists.txt2
-rw-r--r--Source/cmProcessTools.cxx93
-rw-r--r--Source/cmProcessTools.h82
3 files changed, 177 insertions, 0 deletions
diff --git a/Source/CMakeLists.txt b/Source/CMakeLists.txt
index 3180339..964099c 100644
--- a/Source/CMakeLists.txt
+++ b/Source/CMakeLists.txt
@@ -187,6 +187,8 @@ SET(SRCS
cmOrderDirectories.h
cmPolicies.h
cmPolicies.cxx
+ cmProcessTools.cxx
+ cmProcessTools.h
cmProperty.cxx
cmProperty.h
cmPropertyDefinition.cxx
diff --git a/Source/cmProcessTools.cxx b/Source/cmProcessTools.cxx
new file mode 100644
index 0000000..ef40f57
--- /dev/null
+++ b/Source/cmProcessTools.cxx
@@ -0,0 +1,93 @@
+/*=========================================================================
+
+ Program: CMake - Cross-Platform Makefile Generator
+ Module: $RCSfile$
+ Language: C++
+ Date: $Date$
+ Version: $Revision$
+
+ Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved.
+ See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details.
+
+ This software is distributed WITHOUT ANY WARRANTY; without even
+ the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
+ PURPOSE. See the above copyright notices for more information.
+
+=========================================================================*/
+#include "cmProcessTools.h"
+
+#include <cmsys/Process.h>
+
+//----------------------------------------------------------------------------
+void cmProcessTools::RunProcess(struct cmsysProcess_s* cp,
+ OutputParser* out, OutputParser* err)
+{
+ cmsysProcess_Execute(cp);
+ char* data = 0;
+ int length = 0;
+ int p;
+ while((out||err) && (p=cmsysProcess_WaitForData(cp, &data, &length, 0), p))
+ {
+ if(out && p == cmsysProcess_Pipe_STDOUT)
+ {
+ if(!out->Process(data, length))
+ {
+ out = 0;
+ }
+ }
+ else if(err && p == cmsysProcess_Pipe_STDERR)
+ {
+ if(!err->Process(data, length))
+ {
+ err = 0;
+ }
+ }
+ }
+ cmsysProcess_WaitForExit(cp, 0);
+}
+
+
+//----------------------------------------------------------------------------
+cmProcessTools::LineParser::LineParser(char sep, bool ignoreCR):
+ Separator(sep), IgnoreCR(ignoreCR), Log(0), Prefix(0)
+{
+}
+
+//----------------------------------------------------------------------------
+void cmProcessTools::LineParser::SetLog(std::ostream* log, const char* prefix)
+{
+ this->Log = log;
+ this->Prefix = prefix? prefix : "";
+}
+
+//----------------------------------------------------------------------------
+bool cmProcessTools::LineParser::ProcessChunk(const char* first, int length)
+{
+ const char* last = first + length;
+ for(const char* c = first; c != last; ++c)
+ {
+ if(*c == this->Separator)
+ {
+ // Log this line.
+ if(this->Log && this->Prefix)
+ {
+ *this->Log << this->Prefix << this->Line << "\n";
+ }
+
+ // Hand this line to the subclass implementation.
+ if(!this->ProcessLine())
+ {
+ this->Line = "";
+ return false;
+ }
+
+ this->Line = "";
+ }
+ else if(*c != '\r' || !this->IgnoreCR)
+ {
+ // Append this character to the line under construction.
+ this->Line.append(1, *c);
+ }
+ }
+ return true;
+}
diff --git a/Source/cmProcessTools.h b/Source/cmProcessTools.h
new file mode 100644
index 0000000..c438527
--- /dev/null
+++ b/Source/cmProcessTools.h
@@ -0,0 +1,82 @@
+/*=========================================================================
+
+ Program: CMake - Cross-Platform Makefile Generator
+ Module: $RCSfile$
+ Language: C++
+ Date: $Date$
+ Version: $Revision$
+
+ Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved.
+ See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details.
+
+ This software is distributed WITHOUT ANY WARRANTY; without even
+ the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
+ PURPOSE. See the above copyright notices for more information.
+
+=========================================================================*/
+#ifndef cmProcessTools_h
+#define cmProcessTools_h
+
+#include "cmStandardIncludes.h"
+
+/** \class cmProcessTools
+ * \brief Helper classes for process output parsing
+ *
+ */
+class cmProcessTools
+{
+public:
+ /** Abstract interface for process output parsers. */
+ class OutputParser
+ {
+ public:
+ /** Process the given output data from a tool. Processing may be
+ done incrementally. Returns true if the parser is interested
+ in any more data and false if it is done. */
+ bool Process(const char* data, int length)
+ { return this->ProcessChunk(data, length); }
+ protected:
+ /** Implement in a subclass to process a chunk of data. It should
+ return true only if it is interested in more data. */
+ virtual bool ProcessChunk(const char* data, int length) = 0;
+ };
+
+ /** Process output parser that extracts one line at a time. */
+ class LineParser: public OutputParser
+ {
+ public:
+ /** Construct with line separation character and choose whether to
+ ignore carriage returns. */
+ LineParser(char sep = '\n', bool ignoreCR = true);
+
+ /** Configure logging of lines as they are extracted. */
+ void SetLog(std::ostream* log, const char* prefix);
+ protected:
+ char Separator;
+ bool IgnoreCR;
+ std::ostream* Log;
+ const char* Prefix;
+ std::string Line;
+ virtual bool ProcessChunk(const char* data, int length);
+
+ /** Implement in a subclass to process one line of input. It
+ should return true only if it is interested in more data. */
+ virtual bool ProcessLine() = 0;
+ };
+
+ /** Trivial line handler for simple logging. */
+ class OutputLogger: public LineParser
+ {
+ public:
+ OutputLogger(std::ostream& log, const char* prefix = 0)
+ { this->SetLog(&log, prefix); }
+ private:
+ virtual bool ProcessLine() { return true; }
+ };
+
+ /** Run a process and send output to given parsers. */
+ static void RunProcess(struct cmsysProcess_s* cp,
+ OutputParser* out, OutputParser* err = 0);
+};
+
+#endif