|
|
|
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
|
|
file Copyright.txt or https://cmake.org/licensing for details. */
|
|
|
|
#include "cmFortranParser.h"
|
|
|
|
#include "cmFortranLexer.h"
|
|
|
|
#include "cmSystemTools.h"
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <cmConfigure.h>
|
|
|
|
#include <set>
|
|
|
|
#include <stack>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
bool cmFortranParser_s::FindIncludeFile(const char* dir,
|
|
|
|
const char* includeName,
|
|
|
|
std::string& fileName)
|
|
|
|
{
|
|
|
|
// If the file is a full path, include it directly.
|
|
|
|
if (cmSystemTools::FileIsFullPath(includeName)) {
|
|
|
|
fileName = includeName;
|
|
|
|
return cmSystemTools::FileExists(fileName.c_str(), true);
|
|
|
|
}
|
|
|
|
// Check for the file in the directory containing the including
|
|
|
|
// file.
|
|
|
|
std::string fullName = dir;
|
|
|
|
fullName += "/";
|
|
|
|
fullName += includeName;
|
|
|
|
if (cmSystemTools::FileExists(fullName.c_str(), true)) {
|
|
|
|
fileName = fullName;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search the include path for the file.
|
|
|
|
for (std::vector<std::string>::const_iterator i = this->IncludePath.begin();
|
|
|
|
i != this->IncludePath.end(); ++i) {
|
|
|
|
fullName = *i;
|
|
|
|
fullName += "/";
|
|
|
|
fullName += includeName;
|
|
|
|
if (cmSystemTools::FileExists(fullName.c_str(), true)) {
|
|
|
|
fileName = fullName;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
cmFortranParser_s::cmFortranParser_s(std::vector<std::string> const& includes,
|
|
|
|
std::set<std::string> const& defines,
|
|
|
|
cmFortranSourceInfo& info)
|
|
|
|
: IncludePath(includes)
|
|
|
|
, PPDefinitions(defines)
|
|
|
|
, Info(info)
|
|
|
|
{
|
|
|
|
this->InInterface = false;
|
|
|
|
this->InPPFalseBranch = 0;
|
|
|
|
|
|
|
|
// Initialize the lexical scanner.
|
|
|
|
cmFortran_yylex_init(&this->Scanner);
|
|
|
|
cmFortran_yyset_extra(this, this->Scanner);
|
|
|
|
|
|
|
|
// Create a dummy buffer that is never read but is the fallback
|
|
|
|
// buffer when the last file is popped off the stack.
|
|
|
|
YY_BUFFER_STATE buffer =
|
|
|
|
cmFortran_yy_create_buffer(CM_NULLPTR, 4, this->Scanner);
|
|
|
|
cmFortran_yy_switch_to_buffer(buffer, this->Scanner);
|
|
|
|
}
|
|
|
|
|
|
|
|
cmFortranParser_s::~cmFortranParser_s()
|
|
|
|
{
|
|
|
|
cmFortran_yylex_destroy(this->Scanner);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cmFortranParser_FilePush(cmFortranParser* parser, const char* fname)
|
|
|
|
{
|
|
|
|
// Open the new file and push it onto the stack. Save the old
|
|
|
|
// buffer with it on the stack.
|
|
|
|
if (FILE* file = cmsys::SystemTools::Fopen(fname, "rb")) {
|
|
|
|
YY_BUFFER_STATE current = cmFortranLexer_GetCurrentBuffer(parser->Scanner);
|
|
|
|
std::string dir = cmSystemTools::GetParentDirectory(fname);
|
|
|
|
cmFortranFile f(file, current, dir);
|
|
|
|
YY_BUFFER_STATE buffer =
|
|
|
|
cmFortran_yy_create_buffer(CM_NULLPTR, 16384, parser->Scanner);
|
|
|
|
cmFortran_yy_switch_to_buffer(buffer, parser->Scanner);
|
|
|
|
parser->FileStack.push(f);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cmFortranParser_FilePop(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
// Pop one file off the stack and close it. Switch the lexer back
|
|
|
|
// to the next one on the stack.
|
|
|
|
if (parser->FileStack.empty()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
cmFortranFile f = parser->FileStack.top();
|
|
|
|
parser->FileStack.pop();
|
|
|
|
fclose(f.File);
|
|
|
|
YY_BUFFER_STATE current = cmFortranLexer_GetCurrentBuffer(parser->Scanner);
|
|
|
|
cmFortran_yy_delete_buffer(current, parser->Scanner);
|
|
|
|
cmFortran_yy_switch_to_buffer(f.Buffer, parser->Scanner);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int cmFortranParser_Input(cmFortranParser* parser, char* buffer,
|
|
|
|
size_t bufferSize)
|
|
|
|
{
|
|
|
|
// Read from the file on top of the stack. If the stack is empty,
|
|
|
|
// the end of the translation unit has been reached.
|
|
|
|
if (!parser->FileStack.empty()) {
|
|
|
|
cmFortranFile& ff = parser->FileStack.top();
|
|
|
|
FILE* file = ff.File;
|
|
|
|
size_t n = fread(buffer, 1, bufferSize, file);
|
|
|
|
if (n > 0) {
|
|
|
|
ff.LastCharWasNewline = buffer[n - 1] == '\n';
|
|
|
|
} else if (!ff.LastCharWasNewline) {
|
|
|
|
// The file ended without a newline. Inject one so
|
|
|
|
// that the file always ends in an end-of-statement.
|
|
|
|
buffer[0] = '\n';
|
|
|
|
n = 1;
|
|
|
|
ff.LastCharWasNewline = true;
|
|
|
|
}
|
|
|
|
return (int)n;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_StringStart(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
parser->TokenString = "";
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* cmFortranParser_StringEnd(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
return parser->TokenString.c_str();
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_StringAppend(cmFortranParser* parser, char c)
|
|
|
|
{
|
|
|
|
parser->TokenString += c;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_SetInInterface(cmFortranParser* parser, bool in)
|
|
|
|
{
|
|
|
|
if (parser->InPPFalseBranch) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
parser->InInterface = in;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cmFortranParser_GetInInterface(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
return parser->InInterface;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_SetOldStartcond(cmFortranParser* parser, int arg)
|
|
|
|
{
|
|
|
|
parser->OldStartcond = arg;
|
|
|
|
}
|
|
|
|
|
|
|
|
int cmFortranParser_GetOldStartcond(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
return parser->OldStartcond;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_Error(cmFortranParser* parser, const char* msg)
|
|
|
|
{
|
|
|
|
parser->Error = msg ? msg : "unknown error";
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleUse(cmFortranParser* parser, const char* name)
|
|
|
|
{
|
|
|
|
if (!parser->InPPFalseBranch) {
|
|
|
|
parser->Info.Requires.insert(cmSystemTools::LowerCase(name));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleLineDirective(cmFortranParser* parser,
|
|
|
|
const char* filename)
|
|
|
|
{
|
|
|
|
// This is a #line directive naming a file encountered during preprocessing.
|
|
|
|
std::string included = filename;
|
|
|
|
|
|
|
|
// Skip #line directives referencing non-files like
|
|
|
|
// "<built-in>" or "<command-line>".
|
|
|
|
if (included.empty() || included[0] == '<') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fix windows file path separators since our lexer does not
|
|
|
|
// process escape sequences in string literals.
|
|
|
|
cmSystemTools::ReplaceString(included, "\\\\", "\\");
|
|
|
|
cmSystemTools::ConvertToUnixSlashes(included);
|
|
|
|
|
|
|
|
// Save the named file as included in the source.
|
|
|
|
if (cmSystemTools::FileExists(included, true)) {
|
|
|
|
parser->Info.Includes.insert(included);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleInclude(cmFortranParser* parser, const char* name)
|
|
|
|
{
|
|
|
|
if (parser->InPPFalseBranch) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If processing an include statement there must be an open file.
|
|
|
|
assert(!parser->FileStack.empty());
|
|
|
|
|
|
|
|
// Get the directory containing the source in which the include
|
|
|
|
// statement appears. This is always the first search location for
|
|
|
|
// Fortran include files.
|
|
|
|
std::string dir = parser->FileStack.top().Directory;
|
|
|
|
|
|
|
|
// Find the included file. If it cannot be found just ignore the
|
|
|
|
// problem because either the source will not compile or the user
|
|
|
|
// does not care about depending on this included source.
|
|
|
|
std::string fullName;
|
|
|
|
if (parser->FindIncludeFile(dir.c_str(), name, fullName)) {
|
|
|
|
// Found the included file. Save it in the set of included files.
|
|
|
|
parser->Info.Includes.insert(fullName);
|
|
|
|
|
|
|
|
// Parse it immediately to translate the source inline.
|
|
|
|
cmFortranParser_FilePush(parser, fullName.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleModule(cmFortranParser* parser, const char* name)
|
|
|
|
{
|
|
|
|
if (!parser->InPPFalseBranch && !parser->InInterface) {
|
|
|
|
parser->Info.Provides.insert(cmSystemTools::LowerCase(name));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleDefine(cmFortranParser* parser, const char* macro)
|
|
|
|
{
|
|
|
|
if (!parser->InPPFalseBranch) {
|
|
|
|
parser->PPDefinitions.insert(macro);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleUndef(cmFortranParser* parser, const char* macro)
|
|
|
|
{
|
|
|
|
if (!parser->InPPFalseBranch) {
|
|
|
|
std::set<std::string>::iterator match;
|
|
|
|
match = parser->PPDefinitions.find(macro);
|
|
|
|
if (match != parser->PPDefinitions.end()) {
|
|
|
|
parser->PPDefinitions.erase(match);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleIfdef(cmFortranParser* parser, const char* macro)
|
|
|
|
{
|
|
|
|
// A new PP branch has been opened
|
|
|
|
parser->SkipToEnd.push(false);
|
|
|
|
|
|
|
|
if (parser->InPPFalseBranch) {
|
|
|
|
parser->InPPFalseBranch++;
|
|
|
|
} else if (parser->PPDefinitions.find(macro) ==
|
|
|
|
parser->PPDefinitions.end()) {
|
|
|
|
parser->InPPFalseBranch = 1;
|
|
|
|
} else {
|
|
|
|
parser->SkipToEnd.top() = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleIfndef(cmFortranParser* parser, const char* macro)
|
|
|
|
{
|
|
|
|
// A new PP branch has been opened
|
|
|
|
parser->SkipToEnd.push(false);
|
|
|
|
|
|
|
|
if (parser->InPPFalseBranch) {
|
|
|
|
parser->InPPFalseBranch++;
|
|
|
|
} else if (parser->PPDefinitions.find(macro) !=
|
|
|
|
parser->PPDefinitions.end()) {
|
|
|
|
parser->InPPFalseBranch = 1;
|
|
|
|
} else {
|
|
|
|
// ignore other branches
|
|
|
|
parser->SkipToEnd.top() = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleIf(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
/* Note: The current parser is _not_ able to get statements like
|
|
|
|
* #if 0
|
|
|
|
* #if 1
|
|
|
|
* #if MYSMBOL
|
|
|
|
* #if defined(MYSYMBOL)
|
|
|
|
* #if defined(MYSYMBOL) && ...
|
|
|
|
* right. The same for #elif. Thus in
|
|
|
|
* #if SYMBOL_1
|
|
|
|
* ..
|
|
|
|
* #elif SYMBOL_2
|
|
|
|
* ...
|
|
|
|
* ...
|
|
|
|
* #elif SYMBOL_N
|
|
|
|
* ..
|
|
|
|
* #else
|
|
|
|
* ..
|
|
|
|
* #endif
|
|
|
|
* _all_ N+1 branches are considered. If you got something like this
|
|
|
|
* #if defined(MYSYMBOL)
|
|
|
|
* #if !defined(MYSYMBOL)
|
|
|
|
* use
|
|
|
|
* #ifdef MYSYMBOL
|
|
|
|
* #ifndef MYSYMBOL
|
|
|
|
* instead.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// A new PP branch has been opened
|
|
|
|
// Never skip! See note above.
|
|
|
|
parser->SkipToEnd.push(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleElif(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
/* Note: There are parser limitations. See the note at
|
|
|
|
* cmFortranParser_RuleIf(..)
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Always taken unless an #ifdef or #ifndef-branch has been taken
|
|
|
|
// already. If the second condition isn't meet already
|
|
|
|
// (parser->InPPFalseBranch == 0) correct it.
|
|
|
|
if (!parser->SkipToEnd.empty() && parser->SkipToEnd.top() &&
|
|
|
|
!parser->InPPFalseBranch) {
|
|
|
|
parser->InPPFalseBranch = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleElse(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
// if the parent branch is false do nothing!
|
|
|
|
if (parser->InPPFalseBranch > 1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// parser->InPPFalseBranch is either 0 or 1. We change it depending on
|
|
|
|
// parser->SkipToEnd.top()
|
|
|
|
if (!parser->SkipToEnd.empty() && parser->SkipToEnd.top()) {
|
|
|
|
parser->InPPFalseBranch = 1;
|
|
|
|
} else {
|
|
|
|
parser->InPPFalseBranch = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmFortranParser_RuleEndif(cmFortranParser* parser)
|
|
|
|
{
|
|
|
|
if (!parser->SkipToEnd.empty()) {
|
|
|
|
parser->SkipToEnd.pop();
|
|
|
|
}
|
|
|
|
|
|
|
|
// #endif doesn't know if there was a "#else" in before, so it
|
|
|
|
// always decreases InPPFalseBranch
|
|
|
|
if (parser->InPPFalseBranch) {
|
|
|
|
parser->InPPFalseBranch--;
|
|
|
|
}
|
|
|
|
}
|