2018-01-30 11:54:47 -08:00
|
|
|
#ifndef FORTRAN_PRESCAN_H_
|
|
|
|
#define FORTRAN_PRESCAN_H_
|
|
|
|
|
|
|
|
// Defines a fast Fortran source prescanning phase that implements some
|
|
|
|
// character-level features of the language that can be inefficient to
|
|
|
|
// support directly in a backtracking parser. This phase handles Fortran
|
|
|
|
// line continuation, comment removal, card image margins, padding out
|
|
|
|
// fixed form character literals on truncated card images, and drives the
|
|
|
|
// Fortran source preprocessor.
|
|
|
|
|
2018-02-02 15:52:43 -08:00
|
|
|
#include "message.h"
|
2018-01-30 11:54:47 -08:00
|
|
|
#include "preprocessor.h"
|
2018-02-09 14:04:11 -08:00
|
|
|
#include "provenance.h"
|
2018-01-30 11:54:47 -08:00
|
|
|
#include "source.h"
|
|
|
|
#include <optional>
|
2018-02-09 14:04:11 -08:00
|
|
|
#include <string>
|
2018-01-30 11:54:47 -08:00
|
|
|
|
|
|
|
namespace Fortran {
|
2018-02-07 12:04:42 -08:00
|
|
|
namespace parser {
|
2018-01-30 11:54:47 -08:00
|
|
|
|
|
|
|
class Prescanner {
|
2018-02-05 12:54:36 -08:00
|
|
|
public:
|
2018-02-12 12:48:13 -08:00
|
|
|
explicit Prescanner(Messages &messages)
|
|
|
|
: messages_{messages}, preprocessor_{*this} {}
|
2018-02-01 15:01:23 -08:00
|
|
|
|
2018-02-02 15:52:43 -08:00
|
|
|
Messages &messages() const { return messages_; }
|
|
|
|
bool anyFatalErrors() const { return anyFatalErrors_; }
|
2018-01-30 11:54:47 -08:00
|
|
|
|
|
|
|
Prescanner &set_fixedForm(bool yes) {
|
|
|
|
inFixedForm_ = yes;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
Prescanner &set_enableOldDebugLines(bool yes) {
|
|
|
|
enableOldDebugLines_ = yes;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
Prescanner &set_enableBackslashEscapesInCharLiterals(bool yes) {
|
|
|
|
enableBackslashEscapesInCharLiterals_ = yes;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
Prescanner &set_fixedFormColumnLimit(int limit) {
|
|
|
|
fixedFormColumnLimit_ = limit;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2018-02-12 12:48:13 -08:00
|
|
|
const AllSources &allSources() const { return messages_.allSources(); }
|
|
|
|
|
|
|
|
CookedSource Prescan(AllSources *);
|
2018-01-30 11:54:47 -08:00
|
|
|
std::optional<TokenSequence> NextTokenizedLine();
|
2018-02-09 14:04:11 -08:00
|
|
|
Provenance GetCurrentProvenance() const { return GetProvenance(at_); }
|
|
|
|
std::string GetCurrentPath() const; // __FILE__
|
|
|
|
int GetCurrentLineNumber() const; // __LINE__
|
2018-01-30 11:54:47 -08:00
|
|
|
|
2018-02-05 12:54:36 -08:00
|
|
|
private:
|
2018-02-01 12:08:02 -08:00
|
|
|
void BeginSourceLine(const char *at) {
|
2018-01-30 11:54:47 -08:00
|
|
|
at_ = at;
|
2018-02-12 11:56:42 -08:00
|
|
|
column_ = 1;
|
2018-01-30 11:54:47 -08:00
|
|
|
tabInCurrentLine_ = false;
|
|
|
|
preventHollerith_ = false;
|
2018-01-30 15:22:26 -08:00
|
|
|
delimiterNesting_ = 0;
|
2018-01-30 11:54:47 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void BeginSourceLineAndAdvance() {
|
|
|
|
BeginSourceLine(lineStart_);
|
|
|
|
NextLine();
|
|
|
|
}
|
|
|
|
|
2018-02-09 14:04:11 -08:00
|
|
|
Provenance GetProvenance(const char *sourceChar) const {
|
|
|
|
return startProvenance_ + sourceChar - start_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EmitChar(TokenSequence *tokens, char ch) {
|
|
|
|
tokens->PutNextTokenChar(ch, GetCurrentProvenance());
|
|
|
|
}
|
|
|
|
|
2018-01-30 11:54:47 -08:00
|
|
|
char EmitCharAndAdvance(TokenSequence *tokens, char ch) {
|
2018-02-09 14:04:11 -08:00
|
|
|
EmitChar(tokens, ch);
|
2018-01-30 11:54:47 -08:00
|
|
|
NextChar();
|
|
|
|
return *at_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void NextLine();
|
2018-01-30 15:22:26 -08:00
|
|
|
void LabelField(TokenSequence *);
|
2018-01-30 11:54:47 -08:00
|
|
|
void NextChar();
|
|
|
|
void SkipSpaces();
|
|
|
|
bool NextToken(TokenSequence *);
|
|
|
|
bool ExponentAndKind(TokenSequence *);
|
|
|
|
void QuotedCharacterLiteral(TokenSequence *);
|
|
|
|
bool PadOutCharacterLiteral();
|
2018-01-30 15:22:26 -08:00
|
|
|
bool CommentLines();
|
|
|
|
bool CommentLinesAndPreprocessorDirectives();
|
2018-01-30 11:54:47 -08:00
|
|
|
bool IsFixedFormCommentLine(const char *);
|
|
|
|
bool IsFreeFormComment(const char *);
|
|
|
|
bool IsPreprocessorDirectiveLine(const char *);
|
|
|
|
const char *FixedFormContinuationLine();
|
|
|
|
bool FixedFormContinuation();
|
|
|
|
bool FreeFormContinuation();
|
2018-02-09 14:04:11 -08:00
|
|
|
void PayNewlineDebt(CookedSource *);
|
2018-01-30 11:54:47 -08:00
|
|
|
|
2018-02-02 15:52:43 -08:00
|
|
|
Messages &messages_;
|
2018-02-09 14:04:11 -08:00
|
|
|
|
|
|
|
Provenance startProvenance_;
|
|
|
|
const char *start_{nullptr}; // beginning of sourceFile_ content
|
|
|
|
const char *limit_{nullptr}; // first address after end of source
|
2018-01-30 11:54:47 -08:00
|
|
|
const char *at_{nullptr}; // next character to process; < lineStart_
|
|
|
|
int column_{1}; // card image column position of next character
|
2018-02-09 14:04:11 -08:00
|
|
|
const char *lineStart_{nullptr}; // next line to process; <= limit_
|
|
|
|
bool tabInCurrentLine_{false};
|
|
|
|
bool preventHollerith_{false};
|
|
|
|
|
|
|
|
bool anyFatalErrors_{false};
|
2018-01-30 11:54:47 -08:00
|
|
|
int newlineDebt_{0}; // newline characters consumed but not yet emitted
|
|
|
|
bool inCharLiteral_{false};
|
|
|
|
bool inPreprocessorDirective_{false};
|
|
|
|
bool inFixedForm_{true};
|
|
|
|
int fixedFormColumnLimit_{72};
|
|
|
|
bool enableOldDebugLines_{false};
|
|
|
|
bool enableBackslashEscapesInCharLiterals_{true};
|
2018-01-30 15:22:26 -08:00
|
|
|
int delimiterNesting_{0};
|
2018-01-30 11:54:47 -08:00
|
|
|
Preprocessor preprocessor_;
|
|
|
|
};
|
2018-02-07 12:04:42 -08:00
|
|
|
} // namespace parser
|
2018-01-30 11:54:47 -08:00
|
|
|
} // namespace Fortran
|
|
|
|
#endif // FORTRAN_PRESCAN_H_
|