1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157
|
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
file LICENSE.rst or https://cmake.org/licensing for details. */
#pragma once
#include "cmConfigure.h" // IWYU pragma: keep
#include <string>
#include <vector>
#include "cmCTestTestHandler.h"
class cmMakefile;
class cmXMLWriter;
class cmCTest;
/** \class cmCTestMemCheckHandler
* \brief A class that handles ctest -S invocations
*
*/
class cmCTestMemCheckHandler : public cmCTestTestHandler
{
friend class cmCTestRunTest;
public:
using Superclass = cmCTestTestHandler;
void PopulateCustomVectors(cmMakefile* mf) override;
cmCTestMemCheckHandler(cmCTest* ctest);
int GetDefectCount() const;
protected:
int PreProcessHandler() override;
int PostProcessHandler() override;
void GenerateTestCommand(std::vector<std::string>& args, int test) override;
private:
enum
{ // Memory checkers
UNKNOWN = 0,
VALGRIND,
PURIFY,
DRMEMORY,
BOUNDS_CHECKER,
// checkers after here do not use the standard error list
CUDA_SANITIZER,
ADDRESS_SANITIZER,
LEAK_SANITIZER,
THREAD_SANITIZER,
MEMORY_SANITIZER,
UB_SANITIZER
};
public:
enum
{ // Memory faults
ABR = 0,
ABW,
ABWL,
COR,
EXU,
FFM,
FIM,
FMM,
FMR,
FMW,
FUM,
IPR,
IPW,
MAF,
MLK,
MPK,
NPR,
ODS,
PAR,
PLK,
UMC,
UMR,
NO_MEMORY_FAULT
};
private:
enum
{ // Program statuses
NOT_RUN = 0,
TIMEOUT,
SEGFAULT,
ILLEGAL,
INTERRUPT,
NUMERICAL,
OTHER_FAULT,
FAILED,
BAD_COMMAND,
COMPLETED
};
std::string BoundsCheckerDPBDFile;
std::string BoundsCheckerXMLFile;
std::string MemoryTester;
std::vector<std::string> MemoryTesterDynamicOptions;
std::vector<std::string> MemoryTesterOptions;
int MemoryTesterStyle = UNKNOWN;
std::string MemoryTesterOutputFile;
std::string MemoryTesterEnvironmentVariable;
// these are used to store the types of errors that can show up
std::vector<std::string> ResultStrings;
std::vector<std::string> ResultStringsLong;
std::vector<int> GlobalResults;
bool LogWithPID = false; // does log file add pid
int DefectCount = 0;
std::vector<int>::size_type FindOrAddWarning(std::string const& warning);
// initialize the ResultStrings and ResultStringsLong for
// this type of checker
void InitializeResultsVectors();
//! Initialize memory checking subsystem.
bool InitializeMemoryChecking();
/**
* Generate CTest DynamicAnalysis.xml files
*/
void GenerateCTestXML(cmXMLWriter& xml) override;
std::vector<std::string> CustomPreMemCheck;
std::vector<std::string> CustomPostMemCheck;
//! Parse Valgrind/Purify/Bounds Checker result out of the output
// string. After running, log holds the output and results hold the
// different memory errors.
bool ProcessMemCheckOutput(std::string const& str, std::string& log,
std::vector<int>& results);
bool ProcessMemCheckValgrindOutput(std::string const& str, std::string& log,
std::vector<int>& results);
bool ProcessMemCheckDrMemoryOutput(std::string const& str, std::string& log,
std::vector<int>& results);
bool ProcessMemCheckPurifyOutput(std::string const& str, std::string& log,
std::vector<int>& results);
bool ProcessMemCheckCudaOutput(std::string const& str, std::string& log,
std::vector<int>& results);
bool ProcessMemCheckSanitizerOutput(std::string const& str, std::string& log,
std::vector<int>& results);
bool ProcessMemCheckBoundsCheckerOutput(std::string const& str,
std::string& log,
std::vector<int>& results);
void PostProcessTest(cmCTestTestResult& res, int test);
void PostProcessBoundsCheckerTest(cmCTestTestResult& res, int test);
void PostProcessDrMemoryTest(cmCTestTestResult& res, int test);
//! append MemoryTesterOutputFile to the test log
void AppendMemTesterOutput(cmCTestTestHandler::cmCTestTestResult& res,
std::string const& filename);
//! generate the output filename for the given test index
void TestOutputFileNames(int test, std::vector<std::string>& files);
};
|