Commit 2dec1060 authored by albert-github's avatar albert-github

Bug 625601 - FORTRAN: recognition free versus fixed formatted code

The recognition of the type (free or fixed) of Fortran code is not reliable possible. A well known possibility as used with compilers as well is to specify the type of code by means of the extension.
With EXTENSION_MAPPING it is possible to select the type of Fortran code, when not explicitly set doxygen tries to guess the type of Fortran code.
parent 8eeaae0b
...@@ -460,11 +460,18 @@ settings where overruled. ...@@ -460,11 +460,18 @@ settings where overruled.
When using doxygen for Fortran code you should When using doxygen for Fortran code you should
set \ref cfg_optimize_for_fortran "OPTIMIZE_FOR_FORTRAN" to \c YES. set \ref cfg_optimize_for_fortran "OPTIMIZE_FOR_FORTRAN" to \c YES.
The parser tries to guess wheter the source code is fixed format Fortran or
free format Fortran code. This is not always correct, in the later case
one should use \ref cfg_extension_mapping "EXTENSION_MAPPING" to correct this.
By setteing `EXTENSION_MAPPING = f=FortranFixed f90=FortranFree` files with
extension \c f90 are interpreted as fixed format Fortran code and files with
extension \c f are interpreted as free format Fortran code.
For Fortran "!>" or "!<" starts a comment and "!!" or "!>" can be used to For Fortran "!>" or "!<" starts a comment and "!!" or "!>" can be used to
continue an one line comment into a multi-line comment. continue an one line comment into a multi-line comment.
Here is an example of a documented Fortran subroutine: Here is an example of a documented Fortran subroutine:
\verbatim \code{.f}
!> Build the restriction matrix for the aggregation !> Build the restriction matrix for the aggregation
!! method. !! method.
!! @param aggr information about the aggregates !! @param aggr information about the aggregates
...@@ -474,18 +481,20 @@ Here is an example of a documented Fortran subroutine: ...@@ -474,18 +481,20 @@ Here is an example of a documented Fortran subroutine:
Type(SpMtx), intent(in) :: A !< our fine level matrix Type(SpMtx), intent(in) :: A !< our fine level matrix
Type(Aggrs), intent(in) :: aggr Type(Aggrs), intent(in) :: aggr
Type(SpMtx), intent(out) :: Restrict !< Our restriction matrix Type(SpMtx), intent(out) :: Restrict !< Our restriction matrix
\endverbatim !...
end subroutine
\endcode
As an alternative you can also use comments in fixed format code: As an alternative you can also use comments in fixed format code:
\verbatim \code{.f}
C> Function comment C> Function comment
C> another line of comment C> another line of comment
function A(i) function A(i)
C> input parameter C> input parameter
integer i integer i
end function A end function A
\endverbatim \endcode
\subsection tclblocks Comment blocks in Tcl \subsection tclblocks Comment blocks in Tcl
......
...@@ -588,7 +588,10 @@ Go to the <a href="commands.html">next</a> section or return to the ...@@ -588,7 +588,10 @@ Go to the <a href="commands.html">next</a> section or return to the
Doxygen has a built-in mapping, but you can override or extend it using this tag. Doxygen has a built-in mapping, but you can override or extend it using this tag.
The format is <code>ext=language</code>, where \c ext is a file extension, and language is one of The format is <code>ext=language</code>, where \c ext is a file extension, and language is one of
the parsers supported by doxygen: IDL, Java, Javascript, C#, C, C++, D, PHP, the parsers supported by doxygen: IDL, Java, Javascript, C#, C, C++, D, PHP,
Objective-C, Python, Fortran, VHDL. Objective-C, Python, Fortran (fixed format Fortran: FortranFixed,
free formatted Fortran: FortranFree, unknown formatted Fortran: Fortran. In
the later case the parser tries to guess whether the code is fixed or free
formatted code, this is the default for Fortran type files), VHDL.
For instance to make doxygen treat For instance to make doxygen treat
<code>.inc</code> files as Fortran files (default is PHP), and <code>.f</code> files as C (default is Fortran), <code>.inc</code> files as Fortran files (default is PHP), and <code>.f</code> files as C (default is Fortran),
......
...@@ -9880,6 +9880,8 @@ void initDoxygen() ...@@ -9880,6 +9880,8 @@ void initDoxygen()
Doxygen::parserManager->registerParser("c", new CLanguageScanner, TRUE); Doxygen::parserManager->registerParser("c", new CLanguageScanner, TRUE);
Doxygen::parserManager->registerParser("python", new PythonLanguageScanner); Doxygen::parserManager->registerParser("python", new PythonLanguageScanner);
Doxygen::parserManager->registerParser("fortran", new FortranLanguageScanner); Doxygen::parserManager->registerParser("fortran", new FortranLanguageScanner);
Doxygen::parserManager->registerParser("fortranfree", new FortranLanguageScannerFree);
Doxygen::parserManager->registerParser("fortranfixed", new FortranLanguageScannerFixed);
Doxygen::parserManager->registerParser("vhdl", new VHDLLanguageScanner); Doxygen::parserManager->registerParser("vhdl", new VHDLLanguageScanner);
Doxygen::parserManager->registerParser("dbusxml", new DBusXMLScanner); Doxygen::parserManager->registerParser("dbusxml", new DBusXMLScanner);
Doxygen::parserManager->registerParser("tcl", new TclLanguageScanner); Doxygen::parserManager->registerParser("tcl", new TclLanguageScanner);
......
...@@ -28,7 +28,7 @@ void parseFortranCode(CodeOutputInterface &,const char *,const QCString &, ...@@ -28,7 +28,7 @@ void parseFortranCode(CodeOutputInterface &,const char *,const QCString &,
bool ,const char *,FileDef *fd, bool ,const char *,FileDef *fd,
int startLine,int endLine,bool inlineFragment, int startLine,int endLine,bool inlineFragment,
MemberDef *memberDef,bool showLineNumbers,Definition *searchCtx, MemberDef *memberDef,bool showLineNumbers,Definition *searchCtx,
bool collectRefs); bool collectRefs, FortranKind codeType);
void resetFortranCodeParserState(); void resetFortranCodeParserState();
void codeFreeScanner(); void codeFreeScanner();
......
...@@ -154,11 +154,13 @@ static int bracketCount = 0; ...@@ -154,11 +154,13 @@ static int bracketCount = 0;
// simplified way to know if this is fixed form // simplified way to know if this is fixed form
// duplicate in fortranscanner.l // duplicate in fortranscanner.l
static bool recognizeFixedForm(const char* contents) static bool recognizeFixedForm(const char* contents, FortranKind codeType)
{ {
int column=0; int column=0;
bool skipLine=FALSE; bool skipLine=FALSE;
if (codeType == FORTRAN_FIXED) return TRUE;
if (codeType == FORTRAN_FREE) return FALSE;
for (int i=0;;i++) for (int i=0;;i++)
{ {
column++; column++;
...@@ -1108,7 +1110,7 @@ void parseFortranCode(CodeOutputInterface &od,const char *className,const QCStri ...@@ -1108,7 +1110,7 @@ void parseFortranCode(CodeOutputInterface &od,const char *className,const QCStri
bool exBlock, const char *exName,FileDef *fd, bool exBlock, const char *exName,FileDef *fd,
int startLine,int endLine,bool inlineFragment, int startLine,int endLine,bool inlineFragment,
MemberDef *memberDef,bool,Definition *searchCtx, MemberDef *memberDef,bool,Definition *searchCtx,
bool collectXRefs) bool collectXRefs, FortranKind codeType)
{ {
//printf("***parseCode() exBlock=%d exName=%s fd=%p\n",exBlock,exName,fd); //printf("***parseCode() exBlock=%d exName=%s fd=%p\n",exBlock,exName,fd);
...@@ -1122,7 +1124,7 @@ void parseFortranCode(CodeOutputInterface &od,const char *className,const QCStri ...@@ -1122,7 +1124,7 @@ void parseFortranCode(CodeOutputInterface &od,const char *className,const QCStri
g_code = &od; g_code = &od;
g_inputString = s; g_inputString = s;
g_inputPosition = 0; g_inputPosition = 0;
g_isFixedForm = recognizeFixedForm((const char*)s); g_isFixedForm = recognizeFixedForm((const char*)s,codeType);
g_currentFontClass = 0; g_currentFontClass = 0;
g_needsTermination = FALSE; g_needsTermination = FALSE;
g_searchCtx = searchCtx; g_searchCtx = searchCtx;
......
...@@ -19,6 +19,7 @@ ...@@ -19,6 +19,7 @@
#define SCANNER_FORTRAN_H #define SCANNER_FORTRAN_H
#include "parserintf.h" #include "parserintf.h"
#include "util.h"
/** \brief Fortran language parser using state-based lexical scanning. /** \brief Fortran language parser using state-based lexical scanning.
* *
...@@ -27,9 +28,10 @@ ...@@ -27,9 +28,10 @@
class FortranLanguageScanner : public ParserInterface class FortranLanguageScanner : public ParserInterface
{ {
public: public:
virtual ~FortranLanguageScanner() {} FortranLanguageScanner(void) { codeType = FORTRAN_UNKNOWN;}
virtual ~FortranLanguageScanner(void) {}
void startTranslationUnit(const char *) {} void startTranslationUnit(const char *) {}
void finishTranslationUnit() {} void finishTranslationUnit(void) {}
void parseInput(const char *fileName, void parseInput(const char *fileName,
const char *fileBuf, const char *fileBuf,
Entry *root, Entry *root,
...@@ -51,8 +53,22 @@ class FortranLanguageScanner : public ParserInterface ...@@ -51,8 +53,22 @@ class FortranLanguageScanner : public ParserInterface
Definition *searchCtx=0, Definition *searchCtx=0,
bool collectXRefs=TRUE bool collectXRefs=TRUE
); );
void resetCodeParserState(); void resetCodeParserState(void);
void parsePrototype(const char *text); void parsePrototype(const char *text);
FortranKind codeType;
};
class FortranLanguageScannerFree : public FortranLanguageScanner
{
public:
FortranLanguageScannerFree(void) { codeType = FORTRAN_FREE; }
};
class FortranLanguageScannerFixed : public FortranLanguageScanner
{
public:
FortranLanguageScannerFixed(void) { codeType = FORTRAN_FIXED; }
}; };
#endif #endif
...@@ -1313,11 +1313,14 @@ void truncatePrepass(int index) ...@@ -1313,11 +1313,14 @@ void truncatePrepass(int index)
// simplified way to know if this is fixed form // simplified way to know if this is fixed form
// duplicate in fortrancode.l // duplicate in fortrancode.l
static bool recognizeFixedForm(const char* contents) static bool recognizeFixedForm(const char* contents, FortranKind codeType)
{ {
int column=0; int column=0;
bool skipLine=FALSE; bool skipLine=FALSE;
if (codeType == FORTRAN_FIXED) return TRUE;
if (codeType == FORTRAN_FREE) return FALSE;
for(int i=0;;i++) { for(int i=0;;i++) {
column++; column++;
...@@ -2243,7 +2246,7 @@ level--; ...@@ -2243,7 +2246,7 @@ level--;
#endif #endif
static void parseMain(const char *fileName,const char *fileBuf,Entry *rt) static void parseMain(const char *fileName,const char *fileBuf,Entry *rt, FortranKind codeType)
{ {
char *tmpBuf = NULL; char *tmpBuf = NULL;
initParser(); initParser();
...@@ -2263,7 +2266,7 @@ static void parseMain(const char *fileName,const char *fileBuf,Entry *rt) ...@@ -2263,7 +2266,7 @@ static void parseMain(const char *fileName,const char *fileBuf,Entry *rt)
inputFile.setName(fileName); inputFile.setName(fileName);
if (inputFile.open(IO_ReadOnly)) if (inputFile.open(IO_ReadOnly))
{ {
isFixedForm = recognizeFixedForm(fileBuf); isFixedForm = recognizeFixedForm(fileBuf,codeType);
if (isFixedForm) if (isFixedForm)
{ {
...@@ -2342,7 +2345,7 @@ void FortranLanguageScanner::parseInput(const char *fileName, ...@@ -2342,7 +2345,7 @@ void FortranLanguageScanner::parseInput(const char *fileName,
printlex(yy_flex_debug, TRUE, __FILE__, fileName); printlex(yy_flex_debug, TRUE, __FILE__, fileName);
::parseMain(fileName,fileBuf,root); ::parseMain(fileName,fileBuf,root,this->codeType);
printlex(yy_flex_debug, FALSE, __FILE__, fileName); printlex(yy_flex_debug, FALSE, __FILE__, fileName);
} }
...@@ -2365,7 +2368,7 @@ void FortranLanguageScanner::parseCode(CodeOutputInterface & codeOutIntf, ...@@ -2365,7 +2368,7 @@ void FortranLanguageScanner::parseCode(CodeOutputInterface & codeOutIntf,
{ {
::parseFortranCode(codeOutIntf,scopeName,input,isExampleBlock,exampleName, ::parseFortranCode(codeOutIntf,scopeName,input,isExampleBlock,exampleName,
fileDef,startLine,endLine,inlineFragment,memberDef, fileDef,startLine,endLine,inlineFragment,memberDef,
showLineNumbers,searchCtx,collectXRefs); showLineNumbers,searchCtx,collectXRefs,this->codeType);
} }
bool FortranLanguageScanner::needsPreprocessing(const QCString &extension) bool FortranLanguageScanner::needsPreprocessing(const QCString &extension)
......
...@@ -6636,6 +6636,8 @@ g_lang2extMap[] = ...@@ -6636,6 +6636,8 @@ g_lang2extMap[] =
{ "c++", "c", SrcLangExt_Cpp }, { "c++", "c", SrcLangExt_Cpp },
{ "python", "python", SrcLangExt_Python }, { "python", "python", SrcLangExt_Python },
{ "fortran", "fortran", SrcLangExt_Fortran }, { "fortran", "fortran", SrcLangExt_Fortran },
{ "fortranfree", "fortranfree", SrcLangExt_Fortran },
{ "fortranfixed", "fortranfixed", SrcLangExt_Fortran },
{ "vhdl", "vhdl", SrcLangExt_VHDL }, { "vhdl", "vhdl", SrcLangExt_VHDL },
{ "dbusxml", "dbusxml", SrcLangExt_XML }, { "dbusxml", "dbusxml", SrcLangExt_XML },
{ "tcl", "tcl", SrcLangExt_Tcl }, { "tcl", "tcl", SrcLangExt_Tcl },
......
...@@ -414,6 +414,8 @@ QCString externalRef(const QCString &relPath,const QCString &ref,bool href); ...@@ -414,6 +414,8 @@ QCString externalRef(const QCString &relPath,const QCString &ref,bool href);
int nextUtf8CharPosition(const QCString &utf8Str,int len,int startPos); int nextUtf8CharPosition(const QCString &utf8Str,int len,int startPos);
const char *writeUtf8Char(FTextStream &t,const char *s); const char *writeUtf8Char(FTextStream &t,const char *s);
enum FortranKind { FORTRAN_UNKNOWN = 0, FORTRAN_FREE, FORTRAN_FIXED};
/** Data associated with a HSV colored image. */ /** Data associated with a HSV colored image. */
struct ColoredImgDataItem struct ColoredImgDataItem
{ {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment