| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470 | /* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying   file Copyright.txt or https://cmake.org/licensing for details.  */#include "cmListFileCache.h"#include "cmListFileLexer.h"#include "cmMessenger.h"#include "cmOutputConverter.h"#include "cmState.h"#include "cmSystemTools.h"#include "cmake.h"#include "cmConfigure.h"#include <algorithm>#include <assert.h>#include <sstream>struct cmListFileParser{  cmListFileParser(cmListFile* lf, cmListFileBacktrace const& lfbt,                   cmMessenger* messenger, const char* filename);  ~cmListFileParser();  void IssueFileOpenError(std::string const& text) const;  void IssueError(std::string const& text) const;  bool ParseFile();  bool ParseFunction(const char* name, long line);  bool AddArgument(cmListFileLexer_Token* token,                   cmListFileArgument::Delimiter delim);  cmListFile* ListFile;  cmListFileBacktrace Backtrace;  cmMessenger* Messenger;  const char* FileName;  cmListFileLexer* Lexer;  cmListFileFunction Function;  enum  {    SeparationOkay,    SeparationWarning,    SeparationError  } Separation;};cmListFileParser::cmListFileParser(cmListFile* lf,                                   cmListFileBacktrace const& lfbt,                                   cmMessenger* messenger,                                   const char* filename)  : ListFile(lf)  , Backtrace(lfbt)  , Messenger(messenger)  , FileName(filename)  , Lexer(cmListFileLexer_New()){}cmListFileParser::~cmListFileParser(){  cmListFileLexer_Delete(this->Lexer);}void cmListFileParser::IssueFileOpenError(const std::string& text) const{  this->Messenger->IssueMessage(cmake::FATAL_ERROR, text, this->Backtrace);}void cmListFileParser::IssueError(const std::string& text) const{  cmListFileContext lfc;  lfc.FilePath = this->FileName;  lfc.Line = cmListFileLexer_GetCurrentLine(this->Lexer);  cmListFileBacktrace lfbt = this->Backtrace;  lfbt = lfbt.Push(lfc);  this->Messenger->IssueMessage(cmake::FATAL_ERROR, text, lfbt);  cmSystemTools::SetFatalErrorOccured();}bool cmListFileParser::ParseFile(){  // Open the file.  cmListFileLexer_BOM bom;  if (!cmListFileLexer_SetFileName(this->Lexer, this->FileName, &bom)) {    this->IssueFileOpenError("cmListFileCache: error can not open file.");    return false;  }  if (bom == cmListFileLexer_BOM_Broken) {    cmListFileLexer_SetFileName(this->Lexer, CM_NULLPTR, CM_NULLPTR);    this->IssueFileOpenError("Error while reading Byte-Order-Mark. "                             "File not seekable?");    return false;  }  // Verify the Byte-Order-Mark, if any.  if (bom != cmListFileLexer_BOM_None && bom != cmListFileLexer_BOM_UTF8) {    cmListFileLexer_SetFileName(this->Lexer, CM_NULLPTR, CM_NULLPTR);    this->IssueFileOpenError(      "File starts with a Byte-Order-Mark that is not UTF-8.");    return false;  }  // Use a simple recursive-descent parser to process the token  // stream.  bool haveNewline = true;  while (cmListFileLexer_Token* token = cmListFileLexer_Scan(this->Lexer)) {    if (token->type == cmListFileLexer_Token_Space) {    } else if (token->type == cmListFileLexer_Token_Newline) {      haveNewline = true;    } else if (token->type == cmListFileLexer_Token_CommentBracket) {      haveNewline = false;    } else if (token->type == cmListFileLexer_Token_Identifier) {      if (haveNewline) {        haveNewline = false;        if (this->ParseFunction(token->text, token->line)) {          this->ListFile->Functions.push_back(this->Function);        } else {          return false;        }      } else {        std::ostringstream error;        error << "Parse error.  Expected a newline, got "              << cmListFileLexer_GetTypeAsString(this->Lexer, token->type)              << " with text \"" << token->text << "\".";        this->IssueError(error.str());        return false;      }    } else {      std::ostringstream error;      error << "Parse error.  Expected a command name, got "            << cmListFileLexer_GetTypeAsString(this->Lexer, token->type)            << " with text \"" << token->text << "\".";      this->IssueError(error.str());      return false;    }  }  return true;}bool cmListFile::ParseFile(const char* filename, cmMessenger* messenger,                           cmListFileBacktrace const& lfbt){  if (!cmSystemTools::FileExists(filename) ||      cmSystemTools::FileIsDirectory(filename)) {    return false;  }  bool parseError = false;  {    cmListFileParser parser(this, lfbt, messenger, filename);    parseError = !parser.ParseFile();  }  return !parseError;}bool cmListFileParser::ParseFunction(const char* name, long line){  // Ininitialize a new function call.  this->Function = cmListFileFunction();  this->Function.Name = name;  this->Function.Line = line;  // Command name has already been parsed.  Read the left paren.  cmListFileLexer_Token* token;  while ((token = cmListFileLexer_Scan(this->Lexer)) &&         token->type == cmListFileLexer_Token_Space) {  }  if (!token) {    std::ostringstream error;    /* clang-format off */    error << "Unexpected end of file.\n"          << "Parse error.  Function missing opening \"(\".";    /* clang-format on */    this->IssueError(error.str());    return false;  }  if (token->type != cmListFileLexer_Token_ParenLeft) {    std::ostringstream error;    error << "Parse error.  Expected \"(\", got "          << cmListFileLexer_GetTypeAsString(this->Lexer, token->type)          << " with text \"" << token->text << "\".";    this->IssueError(error.str());    return false;  }  // Arguments.  unsigned long lastLine;  unsigned long parenDepth = 0;  this->Separation = SeparationOkay;  while ((lastLine = cmListFileLexer_GetCurrentLine(this->Lexer),          token = cmListFileLexer_Scan(this->Lexer))) {    if (token->type == cmListFileLexer_Token_Space ||        token->type == cmListFileLexer_Token_Newline) {      this->Separation = SeparationOkay;      continue;    }    if (token->type == cmListFileLexer_Token_ParenLeft) {      parenDepth++;      this->Separation = SeparationOkay;      if (!this->AddArgument(token, cmListFileArgument::Unquoted)) {        return false;      }    } else if (token->type == cmListFileLexer_Token_ParenRight) {      if (parenDepth == 0) {        return true;      }      parenDepth--;      this->Separation = SeparationOkay;      if (!this->AddArgument(token, cmListFileArgument::Unquoted)) {        return false;      }      this->Separation = SeparationWarning;    } else if (token->type == cmListFileLexer_Token_Identifier ||               token->type == cmListFileLexer_Token_ArgumentUnquoted) {      if (!this->AddArgument(token, cmListFileArgument::Unquoted)) {        return false;      }      this->Separation = SeparationWarning;    } else if (token->type == cmListFileLexer_Token_ArgumentQuoted) {      if (!this->AddArgument(token, cmListFileArgument::Quoted)) {        return false;      }      this->Separation = SeparationWarning;    } else if (token->type == cmListFileLexer_Token_ArgumentBracket) {      if (!this->AddArgument(token, cmListFileArgument::Bracket)) {        return false;      }      this->Separation = SeparationError;    } else if (token->type == cmListFileLexer_Token_CommentBracket) {      this->Separation = SeparationError;    } else {      // Error.      std::ostringstream error;      error << "Parse error.  Function missing ending \")\".  "            << "Instead found "            << cmListFileLexer_GetTypeAsString(this->Lexer, token->type)            << " with text \"" << token->text << "\".";      this->IssueError(error.str());      return false;    }  }  std::ostringstream error;  cmListFileContext lfc;  lfc.FilePath = this->FileName;  lfc.Line = lastLine;  cmListFileBacktrace lfbt = this->Backtrace;  lfbt = lfbt.Push(lfc);  error << "Parse error.  Function missing ending \")\".  "        << "End of file reached.";  this->Messenger->IssueMessage(cmake::FATAL_ERROR, error.str(), lfbt);  return false;}bool cmListFileParser::AddArgument(cmListFileLexer_Token* token,                                   cmListFileArgument::Delimiter delim){  cmListFileArgument a(token->text, delim, token->line);  this->Function.Arguments.push_back(a);  if (this->Separation == SeparationOkay) {    return true;  }  bool isError = (this->Separation == SeparationError ||                  delim == cmListFileArgument::Bracket);  std::ostringstream m;  cmListFileContext lfc;  lfc.FilePath = this->FileName;  lfc.Line = token->line;  cmListFileBacktrace lfbt = this->Backtrace;  lfbt = lfbt.Push(lfc);  m << "Syntax " << (isError ? "Error" : "Warning") << " in cmake code at "    << "column " << token->column << "\n"    << "Argument not separated from preceding token by whitespace.";  /* clang-format on */  if (isError) {    this->Messenger->IssueMessage(cmake::FATAL_ERROR, m.str(), lfbt);    return false;  }  this->Messenger->IssueMessage(cmake::AUTHOR_WARNING, m.str(), lfbt);  return true;}struct cmListFileBacktrace::Entry : public cmListFileContext{  Entry(cmListFileContext const& lfc, Entry* up)    : cmListFileContext(lfc)    , Up(up)    , RefCount(0)  {    if (this->Up) {      this->Up->Ref();    }  }  ~Entry()  {    if (this->Up) {      this->Up->Unref();    }  }  void Ref() { ++this->RefCount; }  void Unref()  {    if (--this->RefCount == 0) {      delete this;    }  }  Entry* Up;  unsigned int RefCount;};cmListFileBacktrace::cmListFileBacktrace(cmStateSnapshot const& bottom,                                         Entry* up,                                         cmListFileContext const& lfc)  : Bottom(bottom)  , Cur(new Entry(lfc, up)){  assert(this->Bottom.IsValid());  this->Cur->Ref();}cmListFileBacktrace::cmListFileBacktrace(cmStateSnapshot const& bottom,                                         Entry* cur)  : Bottom(bottom)  , Cur(cur){  if (this->Cur) {    assert(this->Bottom.IsValid());    this->Cur->Ref();  }}cmListFileBacktrace::cmListFileBacktrace()  : Bottom()  , Cur(CM_NULLPTR){}cmListFileBacktrace::cmListFileBacktrace(cmStateSnapshot const& snapshot)  : Bottom(snapshot.GetCallStackBottom())  , Cur(CM_NULLPTR){}cmListFileBacktrace::cmListFileBacktrace(cmListFileBacktrace const& r)  : Bottom(r.Bottom)  , Cur(r.Cur){  if (this->Cur) {    assert(this->Bottom.IsValid());    this->Cur->Ref();  }}cmListFileBacktrace& cmListFileBacktrace::operator=(  cmListFileBacktrace const& r){  cmListFileBacktrace tmp(r);  std::swap(this->Cur, tmp.Cur);  std::swap(this->Bottom, tmp.Bottom);  return *this;}cmListFileBacktrace::~cmListFileBacktrace(){  if (this->Cur) {    this->Cur->Unref();  }}cmListFileBacktrace cmListFileBacktrace::Push(std::string const& file) const{  // We are entering a file-level scope but have not yet reached  // any specific line or command invocation within it.  This context  // is useful to print when it is at the top but otherwise can be  // skipped during call stack printing.  cmListFileContext lfc;  lfc.FilePath = file;  return cmListFileBacktrace(this->Bottom, this->Cur, lfc);}cmListFileBacktrace cmListFileBacktrace::Push(  cmListFileContext const& lfc) const{  return cmListFileBacktrace(this->Bottom, this->Cur, lfc);}cmListFileBacktrace cmListFileBacktrace::Pop() const{  assert(this->Cur);  return cmListFileBacktrace(this->Bottom, this->Cur->Up);}cmListFileContext const& cmListFileBacktrace::Top() const{  if (this->Cur) {    return *this->Cur;  }  static cmListFileContext const empty;  return empty;}void cmListFileBacktrace::PrintTitle(std::ostream& out) const{  if (!this->Cur) {    return;  }  cmOutputConverter converter(this->Bottom);  cmListFileContext lfc = *this->Cur;  if (!this->Bottom.GetState()->GetIsInTryCompile()) {    lfc.FilePath = converter.ConvertToRelativePath(      this->Bottom.GetState()->GetSourceDirectory(), lfc.FilePath);  }  out << (lfc.Line ? " at " : " in ") << lfc;}void cmListFileBacktrace::PrintCallStack(std::ostream& out) const{  if (!this->Cur || !this->Cur->Up) {    return;  }  bool first = true;  cmOutputConverter converter(this->Bottom);  for (Entry* i = this->Cur->Up; i; i = i->Up) {    if (i->Name.empty()) {      // Skip this whole-file scope.  When we get here we already will      // have printed a more-specific context within the file.      continue;    }    if (first) {      first = false;      out << "Call Stack (most recent call first):\n";    }    cmListFileContext lfc = *i;    if (!this->Bottom.GetState()->GetIsInTryCompile()) {      lfc.FilePath = converter.ConvertToRelativePath(        this->Bottom.GetState()->GetSourceDirectory(), lfc.FilePath);    }    out << "  " << lfc << "\n";  }}std::ostream& operator<<(std::ostream& os, cmListFileContext const& lfc){  os << lfc.FilePath;  if (lfc.Line) {    os << ":" << lfc.Line;    if (!lfc.Name.empty()) {      os << " (" << lfc.Name << ")";    }  }  return os;}bool operator<(const cmListFileContext& lhs, const cmListFileContext& rhs){  if (lhs.Line != rhs.Line) {    return lhs.Line < rhs.Line;  }  return lhs.FilePath < rhs.FilePath;}bool operator==(const cmListFileContext& lhs, const cmListFileContext& rhs){  return lhs.Line == rhs.Line && lhs.FilePath == rhs.FilePath;}bool operator!=(const cmListFileContext& lhs, const cmListFileContext& rhs){  return !(lhs == rhs);}
 |