/** * Name: f90 * Description: Fortran90 programming language. * Author: David Bowler * * Copyright (C) 2009 Free Software Foundation, Inc. */ /** * Deal with strings enclosed with '...' */ state f90_string_single extends Highlight { /[\']/ { language_print ($0); return; } LANGUAGE_SPECIALS { language_print ($0); } } /** * Deal with strings enclosed with "..." */ state f90_string_double extends Highlight { /[\"]/ { language_print ($0); return; } LANGUAGE_SPECIALS { language_print ($0); } } /** * Deal function/subroutine declarations and subroutine calls: end with ) at end of line or then comment */ state f90_func extends Highlight { /\)[ \t]*$/ { language_print ($0); return; } /(\)[ \t]*)(![a-zA-Z_0-9\,\.\(\)\*\%\: \t]*)/ { language_print ($1); comment_face (true); language_print($2); call (eat_one_line); comment_face (false); return; } LANGUAGE_SPECIALS { language_print ($0); } } /** * Highlight variable declarations */ state f90_new_var_list extends Highlight { /* Catch variable names followed by a comment: 1. Continuation marker present */ /([ \t]*::|[ \t]+)([a-zA-Z_0-9\,\.\(\)\*\%\: \t]+[^\&][ \t]*)(\&[ \t]*)(![a-zA-Z_0-9\,\.\(\)\*\%\: \t]*)/ { language_print ($1); variable_name_face(true); language_print ($2); language_print ($3); variable_name_face(false); comment_face (true); language_print ($4); call (eat_one_line); comment_face (false); } /* Catch variable names followed by a comment: 2. No continuation marker (so return)*/ /([ \t]*::|[ \t]+)([a-zA-Z_0-9\,\.\(\)\*\%\: \t]+[^\&][ \t]*)(![a-zA-Z_0-9\,\.\(\)\*\%\: \t]*)/ { language_print ($1); variable_name_face(true); language_print ($2); variable_name_face(false); comment_face (true); language_print ($3); call (eat_one_line); comment_face (false); return; } /* Is this a specifier ? 1. real(var) ? */ /(\([ \t]*)([a-zA-Z0-9_]+)([ \t]*\))/{ language_print($0); } /* Is this a specifier ? 2. real(kind=var) */ /(\([ \t]*)(len|kind)([a-zA-Z0-9_ =]+)(\))/{ language_print($1); keyword_face(true); language_print($2); keyword_face(false); language_print($3); language_print($4); } /* Is this a specifier ? 3. real(kind=selected_real_kind(6,90)) */ /(\([ \t]*)(len|kind)([ \t]*=[ \t]*)(selected_(int_kind|real_kind))([ \t]*\([ \t]*[0-9\,]+[ \t]*\)[ \t]*)(\))/{ language_print($1); keyword_face(true); language_print($2); keyword_face(false); language_print($3); keyword_face(true); language_print($4); keyword_face(false); language_print($6); language_print($7); } /* Highlight modifiers (build-re '(allocatable Allocatable ALLOCATABLE external External EXTERNAL intent Intent INTENT optional Optional OPTIONAL parameter Parameter PARAMETER pointer Pointer POINTER private Private PRIVATE public Public PUBLIC save SAVE Save target TARGET Target)) */ /(\,[ \t]*)(A(LLOCATABLE|llocatable)|E(XTERNAL|xternal)|I(NTENT|ntent)\ |O(PTIONAL|ptional)\ |P(ARAMETER|OINTER|RIVATE|UBLIC|arameter|ointer|rivate|ublic)\ |S(AVE|ave)|T(ARGET|arget)|allocatable|external|intent|optional\ |p(arameter|ointer|rivate|ublic)|save|target)/ { language_print($1); keyword_face(true); language_print($2); keyword_face(false); } /(\,[ \t]*)(D(IMENSION|imension)|dimension)([ \t]*\([ \:\,\-+*a-zA-Z_0-9]+[ \t]*\))/ { language_print($1); keyword_face(true); language_print($2); keyword_face(false); language_print($4); } /* Highlight variable names up to continuation marker */ /([ \t]*::|[^\,\(][ \t]*)([a-zA-Z_0-9]+[a-zA-Z_0-9\,\.\(\)\*\%\:\+\- \t]+[\&][ \t]*)$/ { language_print ($1); variable_name_face(true); language_print ($2); variable_name_face(false); } /* Highlight variable names up to end of line (no continuation marker: return) */ /([ \t]*::|[^\,\(][ \t]*)([a-zA-Z_0-9]+[a-zA-Z_0-9\,\.\(\)\*\%\:\+\- \t]*[^\&][ \t]*)$/ { language_print ($1); variable_name_face(true); language_print ($2); variable_name_face(false); return; } /* Highlight variable names up to equals sign (return after equals)*/ /([ \t]*::|[^\,\(][ \t]*)([a-zA-Z_0-9]+[a-zA-Z_0-9\,\.\(\)\*\%\:\+\- \t]*[^\&])([ \t]*=)/ { language_print ($1); variable_name_face(true); language_print ($2); variable_name_face(false); language_print ($3); return; } LANGUAGE_SPECIALS { language_print ($0); } } /** * Highlight F90 io statements */ state f90_io extends Highlight { /* Catch comments */ /[!]/ { comment_face (true); language_print ($0); call (eat_one_line); comment_face (false); } /* String constants. */ /[\'][^\)]/ { string_face (true); language_print ($0); call (f90_string_single); string_face (false); } /[\"][^\)]/ { string_face (true); language_print ($0); call (f90_string_double); string_face (false); } /* This terminates an io statement */ /\)[^\'\"]/ { language_print ($0); return; } /* IO Keywords. (build-re '(FMT UNIT REC END ERR FILE STATUS ACCESS FORM RECL BLANK IOSTAT EXIST OPENED NUMBER NAME SEQUENTIAL DIRECT FORMATTED UNFORMATTED NEXTREC)) */ /\b(ACCESS|BLANK|DIRECT|E(ND|RR|XIST)|F(ILE|MT|ORM(|ATTED))|IOSTAT\ |N(AME|EXTREC|UMBER)|OPENED|REC(|L)|S(EQUENTIAL|TATUS)\ |UN(FORMATTED|IT))\b/ { keyword_face (true); language_print ($0); keyword_face (false); } /* IO Keywords. (build-re '(fmt unit rec end err file status access form recl blank iostat exist opened number name sequential direct formatted unformatted nextrec)) */ /\b((a|A)ccess|(b|B)lank|(d|D)irect|(e|E)(nd|rr|xist)|(f|F)(ile|mt|orm(|atted))|(i|I)ostat\ |(n|N)(ame|extrec|umber)|(o|O)pened|(r|R)ec(|l)|(s|S)(equential|tatus)\ |(u|U)n(formatted|it))\b/ { keyword_face (true); language_print ($0); keyword_face (false); } LANGUAGE_SPECIALS { language_print ($0); } } state f90 extends HighlightEntry { BEGIN { header (); } END { trailer (); } /* String constants. */ /[\']/ { string_face (true); language_print ($0); call (f90_string_single); string_face (false); } /[\"]/ { string_face (true); language_print ($0); call (f90_string_double); string_face (false); } /* Labels - whitespace followed by number at start of line */ /^[ \t]*[0-9]+/{ keyword_face(true); language_print ($0); keyword_face(false); } /* Comments. We'll only have free-form, modern f90 statements - ! to end of line*/ /[!]/ { comment_face (true); language_print ($0); call (eat_one_line); comment_face (false); } /* builtins - maths, matrices etc */ /* Builtins. (build-re '(abs achar acos adjustl adjustr aimag aint all allocated anint any asin associated atan atan2 bit_size btest ceiling char cmplx conjg cos cosh count cshift date_and_time dble digits dim dot_product dprod eoshift epsilon exp exponent floor fraction huge iachar iand ibclr ibits ibset ichar ieor index int ior ishft ishftc kind lbound len len_trim lge lgt lle llt log logical log10 matmul max maxexponent maxloc maxval merge min minexponent minloc minval mod modulo mvbits nearest nint not pack precision present product radix random_number random_seed range real repeat reshape rrspacing scale scan selected_int_kind selected_real_kind set_exponent shape sign sin sinh size spacing spread sqrt sum system_clock tan tanh tiny transfer transpose trim ubound unpack verify)) */ /\b((a|A)(bs|c(har|os)|djust(l|r)|i(mag|nt)|ll(|ocated)|n(int|y)|s(in|sociated)\ |tan(|2))\ |(b|B)(it_size|test)|(c|C)(eiling|har|mplx|o(njg|s(|h)|unt)|shift)\ |(d|D)(ate_and_time|ble|i(gits|m)|ot_product|prod)\ |(e|E)(oshift|psilon|xp(|onent))|(f|F)(loor|raction)|(h|H)uge\ |(i|I)(a(char|nd)|b(clr|its|set)|char|eor|n(dex|t)|or|shft(|c))|(k|K)ind\ |(l|L)(bound|en(|_trim)|g(e|t)|l(e|t)|og(|10|ical))\ |(m|M)(a(tmul|x(|exponent|loc|val))|erge|in(|exponent|loc|val)|od(|ulo)\ |vbits)\ |(n|N)(earest|int|ot)|(p|P)(ack|r(e(cision|sent)|oduct))\ |(r|R)(a(dix|n(dom_(number|seed)|ge))|e(al|peat|shape)|rspacing)\ |(s|S)(ca(le|n)|e(lected_(int_kind|real_kind)|t_exponent)|hape\ |i(gn|n(|h)|ze)|p(acing|read)|qrt|um|ystem_clock)\ |(t|T)(an(|h)|iny|r(ans(fer|pose)|im))|(u|U)(bound|npack)|(v|V)erify)\b/ { builtin_face (true); language_print ($0); builtin_face (false); } /* Builtins. (build-re '(ABS ACHAR ACOS ADJUSTL ADJUSTR AIMAG AINT ALL ALLOCATED ANINT ANY ASIN ASSOCIATED ATAN ATAN2 BIT_SIZE BTEST CEILING CHAR CMPLX CONJG COS COSH COUNT CSHIFT DATE_AND_TIME DBLE DIGITS DIM DOT_PRODUCT DPROD EOSHIFT EPSILON EXP EXPONENT FLOOR FRACTION HUGE IACHAR IAND IBCLR IBITS IBSET ICHAR IEOR INDEX INT IOR ISHFT ISHFTC KIND LBOUND LEN LEN_TRIM LGE LGT LLE LLT LOG LOGICAL LOG10 MATMUL MAX MAXEXPONENT MAXLOC MAXVAL MERGE MIN MINEXPONENT MINLOC MINVAL MOD MODULO MVBITS NEAREST NINT NOT PACK PRECISION PRESENT PRODUCT RADIX RANDOM_NUMBER RANDOM_SEED RANGE REAL REPEAT RESHAPE RRSPACING SCALE SCAN SELECTED_INT_KIND SELECTED_REAL_KIND SET_EXPONENT SHAPE SIGN SIN SINH SIZE SPACING SPREAD SQRT SUM SYSTEM_CLOCK TAN TANH TINY TRANSFER TRANSPOSE TRIM UBOUND UNPACK VERIFY)) */ /\b(A(BS|C(HAR|OS)|DJUST(L|R)|I(MAG|NT)|LL(|OCATED)|N(INT|Y)|S(IN|SOCIATED)\ |TAN(|2))\ |B(IT_SIZE|TEST)|C(EILING|HAR|MPLX|O(NJG|S(|H)|UNT)|SHIFT)\ |D(ATE_AND_TIME|BLE|I(GITS|M)|OT_PRODUCT|PROD)\ |E(OSHIFT|PSILON|XP(|ONENT))|F(LOOR|RACTION)|HUGE\ |I(A(CHAR|ND)|B(CLR|ITS|SET)|CHAR|EOR|N(DEX|T)|OR|SHFT(|C))|KIND\ |L(BOUND|EN(|_TRIM)|G(E|T)|L(E|T)|OG(|10|ICAL))\ |M(A(TMUL|X(|EXPONENT|LOC|VAL))|ERGE|IN(|EXPONENT|LOC|VAL)|OD(|ULO)\ |VBITS)\ |N(EAREST|INT|OT)|P(ACK|R(E(CISION|SENT)|ODUCT))\ |R(A(DIX|N(DOM_(NUMBER|SEED)|GE))|E(AL|PEAT|SHAPE)|RSPACING)\ |S(CA(LE|N)|E(LECTED_(INT_KIND|REAL_KIND)|T_EXPONENT)|HAPE\ |I(GN|N(|H)|ZE)|P(ACING|READ)|QRT|UM|YSTEM_CLOCK)\ |T(AN(|H)|INY|R(ANS(FER|POSE)|IM))|U(BOUND|NPACK)|VERIFY)\b/ { builtin_face (true); language_print ($0); builtin_face (false); } LANGUAGE_SPECIALS { language_print ($0); } /* Comparators. We have to roll by hand because of the dots - "\b" doesn't delimit here. */ /\.((a|A)nd|(e|E)qv?|(g|G)(e|t)|(l|L)(e|t)|(n|N)e(qv)?|(n|N)ot|(o|O)r|(t|T)rue|(f|F)alse)\./ { keyword_face (true); language_print ($0); keyword_face (false); } /* Comparators. We have to roll by hand because of the dots - "\b" doesn't delimit here. */ /\.(AND|EQV?|G(E|T)|L(E|T)|NE(QV)?|NOT|OR|TRUE|FALSE)\./ { keyword_face (true); language_print ($0); keyword_face (false); } /* function, subroutine declaration or subroutine call: 1. with arguments*/ /(^[ \t]*((c|C)all|(f|F)unction|(s|S)ubroutine)[ \t]+)([a-zA-Z_0-9]+)([ \t]*\()/ { keyword_face(true); language_print($1); keyword_face(false); function_name_face(true); language_print($6); function_name_face(false); language_print($7); call (f90_func); } /* function, subroutine declaration or subroutine call: 1. without arguments*/ /(^[ \t]*((c|C)all|(f|F)unction|(s|S)ubroutine)[ \t]+)([a-zA-Z_0-9]+[ \t]*)$/ { keyword_face(true); language_print($1); keyword_face(false); function_name_face(true); language_print($6); function_name_face(false); language_print($7); } /* function, subroutine declaration or subroutine call*/ /((CALL|FUNCTION|SUBROUTINE)[ \t]+)([a-zA-Z_0-9]+)([ \t]*\()/ { keyword_face(true); language_print($1); keyword_face(false); function_name_face(true); language_print($3); function_name_face(false); language_print($4); call (f90_func); } /* end function, subroutine declaration or subroutine call*/ /(((e|E)nd)[ \t]*)(((c|C)all|(f|F)unction|(s|S)ubroutine)[ \t]+)([a-zA-Z_0-9]+)/ { keyword_face(true); language_print($1); language_print($4); keyword_face(false); function_name_face(true); language_print($9); function_name_face(false); } /* end function, subroutine declaration or subroutine call*/ /((END)[ \t]*)((CALL|FUNCTION|SUBROUTINE)[ \t]+)([a-zA-Z_0-9]+)/ { keyword_face(true); language_print($1); language_print($3); keyword_face(false); function_name_face(true); language_print($5); function_name_face(false); } /* Module, program, use declaration */ /(((e|E)nd)?[ \t]*)(((m|M)odule|(p|P)rogram|(u|U)se)[ \t]+)([a-zA-Z_0-9]+)/ { keyword_face(true); language_print($1); language_print($4); keyword_face(false); function_name_face(true); language_print($9); function_name_face(false); } /* Module, program, use declaration */ /((END)?[ \t]*)((MODULE|PROGRAM|USE)[ \t]+)([a-zA-Z_0-9]+)/ { debug(concat("Strings: ",$0)); debug(concat($1,"|")); debug(concat($2,"|")); debug(concat($3,"|")); debug(concat($4,"|")); debug(concat($5,"|")); debug(concat($6,"|")); keyword_face(true); language_print($1); language_print($3); keyword_face(false); function_name_face(true); language_print($5); function_name_face(false); } /* Function call */ /* Unfortunately, as F90 uses round brackets for function calls and arrays, this breaks */ /* /(=[ \t]*)([a-zA-Z_0-9]+)([ \t]*\()/{ language_print($1); function_name_face(true); language_print($2); function_name_face(false); language_print($3); }*/ /* Variable declaration */ /^([ \t]*)((i|I)nteger|(r|R)eal|(c|C)omplex|(c|C)haracter|(l|L)ogical|([ \t]*(e|E)nd[ \t]*)?(t|T)ype)/ { type_face(true); language_print($0); type_face(false); call (f90_new_var_list); } /^([ \t]*)(INTEGER|REAL|COMPLEX|CHARACTER|LOGICAL|([ \t]*END[ \t]*)?TYPE)/ { type_face(true); language_print($0); type_face(false); call (f90_new_var_list); } /* none */ /\bnone\b/ { type_face(true); language_print($0); type_face(false); } /* IO Statement (build-re '(open close read write inquire backspace endfile rewind )) */ /\b((b|B)ackspace|(c|C)lose|(e|E)ndfile|(i|I)nquire|(o|O)pen|(r|R)e(ad|wind)|(w|W)rite)\b/ { keyword_face (true); language_print ($0); keyword_face (false); call (f90_io); } /* IO Statement (build-re '(OPEN CLOSE READ WRITE INQUIRE BACKSPACE ENDFILE REWIND )) */ /\b(BACKSPACE|CLOSE|ENDFILE|INQUIRE|OPEN|RE(AD|WIND)|WRITE)\b/ { keyword_face (true); language_print ($0); keyword_face (false); call (f90_io); } /* Keywords */ /* (build-re '(allocate allocatable assign assignment block case common contains continue cycle data deallocate dimension do double else elseif elsewhere end enddo endif entry equivalence exit external forall format goto if implicit include intent interface intrinsic module namelist none nullify only operator optional parameter pause pointer precision print private procedure program public recursive result return save select sequence stop subroutine target then use where while)) */ /\b((a|A)(llocat(able|e)|ssign(|ment))|(b|B)lock\ |(c|C)(ase|o(mmon|nt(ains|inue))|ycle)|(d|D)(ata|eallocate|imension|o(|uble))\ |(e|E)(lse(|if|where)|n(d(|do|if)|try)|quivalence|x(it|ternal))\ |(f|F)or(all|mat)|(g|G)oto|(i|I)(f|mplicit|n(clude|t(e(nt|rface)|rinsic)))\ |(m|M)odule\ |(n|N)(amelist|ullify)|(o|O)(nly|p(erator|tional))\ |(p|P)(a(rameter|use)|ointer|r(ecision|i(nt|vate)|o(cedure|gram))|ublic)\ |(r|R)e(cursive|sult|turn)|(s|S)(ave|e(lect|quence)|top|ubroutine)\ |(t|T)(arget|hen)|(u|U)se|(w|W)h(ere|ile))\b/ { keyword_face (true); language_print ($0); keyword_face (false); } /* (build-re '(ALLOCATE ALLOCATABLE ASSIGN ASSIGNMENT BLOCK CASE COMMON CONTAINS CONTINUE CYCLE DATA DEALLOCATE DIMENSION DO DOUBLE ELSE ELSEIF ELSEWHERE END ENDDO ENDIF ENTRY EQUIVALENCE EXIT EXTERNAL FORALL FORMAT GOTO IF IMPLICIT INCLUDE INTENT INTERFACE INTRINSIC MODULE NAMELIST NULLIFY ONLY OPERATOR OPTIONAL PARAMETER PAUSE POINTER PRECISION PRINT PRIVATE PROCEDURE PROGRAM PUBLIC RECURSIVE RESULT RETURN SAVE SELECT SEQUENCE STOP SUBROUTINE TARGET THEN USE WHERE WHILE)) */ /\b(A(LLOCAT(ABLE|E)|SSIGN(|MENT))|BLOCK\ |C(ASE|O(MMON|NT(AINS|INUE))|YCLE)|D(ATA|EALLOCATE|IMENSION|O(|UBLE))\ |E(LSE(|IF|WHERE)|N(D(|DO|IF)|TRY)|QUIVALENCE|X(IT|TERNAL))\ |FOR(ALL|MAT)|GOTO|I(F|MPLICIT|N(CLUDE|T(E(NT|RFACE)|RINSIC)))\ |MODULE\ |N(AMELIST|ULLIFY)|O(NLY|P(ERATOR|TIONAL))\ |P(A(RAMETER|USE)|OINTER|R(ECISION|I(NT|VATE)|O(CEDURE|GRAM))|UBLIC)\ |RE(CURSIVE|SULT|TURN)|S(AVE|E(LECT|QUENCE)|TOP|UBROUTINE)\ |T(ARGET|HEN)|USE|WH(ERE|ILE))\b/ { keyword_face (true); language_print ($0); keyword_face (false); } LANGUAGE_SPECIALS { language_print ($0); } } /* Local variables: mode: c End: */