2023-02-07 22:02:12 +01:00
|
|
|
/*
|
|
|
|
* Cppcheck - A tool for static C/C++ code analysis
|
2023-06-21 16:41:22 +02:00
|
|
|
* Copyright (C) 2007-2023 Cppcheck team.
|
2023-02-07 22:02:12 +01:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "helpers.h"
|
|
|
|
|
2023-08-22 15:25:28 +02:00
|
|
|
#include "filelister.h"
|
2023-02-07 22:02:12 +01:00
|
|
|
#include "path.h"
|
2023-08-22 15:25:28 +02:00
|
|
|
#include "pathmatch.h"
|
2023-04-22 10:22:00 +02:00
|
|
|
#include "preprocessor.h"
|
2023-02-07 22:02:12 +01:00
|
|
|
|
2023-08-02 10:36:17 +02:00
|
|
|
#include <cerrno>
|
2023-03-02 21:50:14 +01:00
|
|
|
#include <cstdio>
|
2023-05-03 17:32:28 +02:00
|
|
|
#include <iostream>
|
2023-08-02 10:36:17 +02:00
|
|
|
#include <fstream> // IWYU pragma: keep
|
|
|
|
#include <list>
|
2023-08-22 15:25:28 +02:00
|
|
|
#include <map>
|
2023-03-02 21:50:14 +01:00
|
|
|
#include <stdexcept>
|
|
|
|
#include <utility>
|
2023-04-30 07:33:19 +02:00
|
|
|
#include <vector>
|
2023-03-02 21:50:14 +01:00
|
|
|
|
2023-02-07 22:02:12 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <windows.h>
|
|
|
|
#else
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2023-04-22 10:22:00 +02:00
|
|
|
#include <simplecpp.h>
|
|
|
|
|
2023-04-30 07:33:19 +02:00
|
|
|
class Suppressions;
|
|
|
|
|
2023-08-22 15:25:28 +02:00
|
|
|
// TODO: better path-only usage
|
2023-02-07 22:02:12 +01:00
|
|
|
ScopedFile::ScopedFile(std::string name, const std::string &content, std::string path)
|
|
|
|
: mName(std::move(name))
|
|
|
|
, mPath(Path::toNativeSeparators(std::move(path)))
|
|
|
|
, mFullPath(Path::join(mPath, mName))
|
|
|
|
{
|
|
|
|
if (!mPath.empty() && mPath != Path::getCurrentPath()) {
|
2023-12-05 20:39:26 +01:00
|
|
|
if (Path::isDirectory(mPath))
|
|
|
|
throw std::runtime_error("ScopedFile(" + mFullPath + ") - directory already exists");
|
2023-02-07 22:02:12 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
if (!CreateDirectoryA(mPath.c_str(), nullptr))
|
|
|
|
throw std::runtime_error("ScopedFile(" + mFullPath + ") - could not create directory");
|
|
|
|
#else
|
|
|
|
if (mkdir(mPath.c_str(), S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH) != 0)
|
|
|
|
throw std::runtime_error("ScopedFile(" + mFullPath + ") - could not create directory");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2023-12-05 20:39:26 +01:00
|
|
|
if (Path::isFile(mFullPath))
|
|
|
|
throw std::runtime_error("ScopedFile(" + mFullPath + ") - file already exists");
|
|
|
|
|
2023-02-07 22:02:12 +01:00
|
|
|
std::ofstream of(mFullPath);
|
|
|
|
if (!of.is_open())
|
|
|
|
throw std::runtime_error("ScopedFile(" + mFullPath + ") - could not open file");
|
|
|
|
of << content;
|
|
|
|
}
|
|
|
|
|
|
|
|
ScopedFile::~ScopedFile() {
|
2023-05-03 17:32:28 +02:00
|
|
|
const int remove_res = std::remove(mFullPath.c_str());
|
|
|
|
if (remove_res != 0) {
|
2023-08-22 15:25:28 +02:00
|
|
|
std::cout << "ScopedFile(" << mFullPath + ") - could not delete file (" << remove_res << ")" << std::endl;
|
2023-05-03 17:32:28 +02:00
|
|
|
}
|
2023-02-07 22:02:12 +01:00
|
|
|
if (!mPath.empty() && mPath != Path::getCurrentPath()) {
|
2023-08-22 15:25:28 +02:00
|
|
|
// TODO: remove all files
|
|
|
|
// TODO: simplify the function call
|
|
|
|
// hack to be able to delete *.plist output files
|
2023-11-07 21:21:24 +01:00
|
|
|
std::list<std::pair<std::string, std::size_t>> files;
|
2023-08-22 15:25:28 +02:00
|
|
|
const std::string res = FileLister::addFiles(files, mPath, {".plist"}, false, PathMatch({}));
|
|
|
|
if (!res.empty()) {
|
|
|
|
std::cout << "ScopedFile(" << mPath + ") - generating file list failed (" << res << ")" << std::endl;
|
|
|
|
}
|
|
|
|
for (const auto &f : files)
|
|
|
|
{
|
|
|
|
const std::string &file = f.first;
|
|
|
|
const int rm_f_res = std::remove(file.c_str());
|
|
|
|
if (rm_f_res != 0) {
|
|
|
|
std::cout << "ScopedFile(" << mPath + ") - could not delete '" << file << "' (" << rm_f_res << ")" << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-07 22:02:12 +01:00
|
|
|
#ifdef _WIN32
|
2023-05-03 17:32:28 +02:00
|
|
|
if (!RemoveDirectoryA(mPath.c_str())) {
|
2023-08-22 15:25:28 +02:00
|
|
|
std::cout << "ScopedFile(" << mFullPath + ") - could not delete folder (" << GetLastError() << ")" << std::endl;
|
2023-05-03 17:32:28 +02:00
|
|
|
}
|
2023-02-07 22:02:12 +01:00
|
|
|
#else
|
2023-05-03 17:32:28 +02:00
|
|
|
const int rmdir_res = rmdir(mPath.c_str());
|
|
|
|
if (rmdir_res == -1) {
|
|
|
|
const int err = errno;
|
2023-08-22 15:25:28 +02:00
|
|
|
std::cout << "ScopedFile(" << mFullPath + ") - could not delete folder (" << err << ")" << std::endl;
|
2023-05-03 17:32:28 +02:00
|
|
|
}
|
2023-02-07 22:02:12 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2023-04-22 10:22:00 +02:00
|
|
|
|
|
|
|
// TODO: we should be using the actual Preprocessor implementation
|
|
|
|
std::string PreprocessorHelper::getcode(Preprocessor &preprocessor, const std::string &filedata, const std::string &cfg, const std::string &filename, Suppressions *inlineSuppression)
|
|
|
|
{
|
|
|
|
simplecpp::OutputList outputList;
|
|
|
|
std::vector<std::string> files;
|
|
|
|
|
|
|
|
std::istringstream istr(filedata);
|
|
|
|
simplecpp::TokenList tokens1(istr, files, Path::simplifyPath(filename), &outputList);
|
|
|
|
if (inlineSuppression)
|
|
|
|
preprocessor.inlineSuppressions(tokens1, *inlineSuppression);
|
|
|
|
tokens1.removeComments();
|
|
|
|
preprocessor.simplifyPragmaAsm(&tokens1);
|
|
|
|
preprocessor.removeComments();
|
|
|
|
preprocessor.setDirectives(tokens1);
|
|
|
|
|
|
|
|
preprocessor.reportOutput(outputList, true);
|
|
|
|
|
|
|
|
if (Preprocessor::hasErrors(outputList))
|
|
|
|
return "";
|
|
|
|
|
|
|
|
std::string ret;
|
|
|
|
try {
|
2023-09-20 14:45:44 +02:00
|
|
|
// TODO: also preserve location information when #include exists - enabling that will fail since #line is treated like a regular token
|
2023-04-22 10:22:00 +02:00
|
|
|
ret = preprocessor.getcode(tokens1, cfg, files, filedata.find("#file") != std::string::npos);
|
|
|
|
} catch (const simplecpp::Output &) {
|
|
|
|
ret.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Since "files" is a local variable the tracking info must be cleared..
|
|
|
|
preprocessor.mMacroUsage.clear();
|
|
|
|
preprocessor.mIfCond.clear();
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2023-10-08 11:29:52 +02:00
|
|
|
|
|
|
|
void PreprocessorHelper::preprocess(const char code[], std::vector<std::string> &files, Tokenizer& tokenizer)
|
|
|
|
{
|
|
|
|
// Raw Tokens..
|
|
|
|
std::istringstream istr(code);
|
|
|
|
const simplecpp::TokenList tokens1(istr, files, files[0]);
|
|
|
|
|
|
|
|
// Preprocess..
|
|
|
|
simplecpp::TokenList tokens2(files);
|
|
|
|
std::map<std::string, simplecpp::TokenList*> filedata;
|
|
|
|
simplecpp::preprocess(tokens2, tokens1, files, filedata, simplecpp::DUI());
|
|
|
|
|
|
|
|
// Tokenizer..
|
|
|
|
tokenizer.createTokens(std::move(tokens2));
|
|
|
|
}
|
|
|
|
|
|
|
|
void PreprocessorHelper::preprocess(Preprocessor &preprocessor, const char code[], std::vector<std::string> &files, Tokenizer& tokenizer)
|
2023-10-21 17:21:12 +02:00
|
|
|
{
|
|
|
|
preprocess(preprocessor, code, files, tokenizer, simplecpp::DUI());
|
|
|
|
}
|
|
|
|
|
|
|
|
void PreprocessorHelper::preprocess(Preprocessor &preprocessor, const char code[], std::vector<std::string> &files, Tokenizer& tokenizer, const simplecpp::DUI& dui)
|
2023-10-08 11:29:52 +02:00
|
|
|
{
|
|
|
|
std::istringstream istr(code);
|
|
|
|
const simplecpp::TokenList tokens1(istr, files, files[0]);
|
|
|
|
|
|
|
|
// Preprocess..
|
|
|
|
simplecpp::TokenList tokens2(files);
|
|
|
|
std::map<std::string, simplecpp::TokenList*> filedata;
|
2023-10-21 17:21:12 +02:00
|
|
|
simplecpp::preprocess(tokens2, tokens1, files, filedata, dui);
|
2023-10-08 11:29:52 +02:00
|
|
|
|
|
|
|
// Tokenizer..
|
|
|
|
tokenizer.createTokens(std::move(tokens2));
|
|
|
|
|
|
|
|
preprocessor.setDirectives(tokens1);
|
|
|
|
}
|