We’re just trying to puzzle this out in fortitude.
Looking at the 2023 standard on deleted and obsolescent features, we have this clause:
B.3.8
CHARACTER*
form of CHARACTER declaration
In addition to theCHARACTER*char-length
form introduced in Fortran 77, Fortran 90 provided theCHARACTER([ LEN = ] type-param-value)
form. The older form(CHARACTER*char-length)
is redundant.
If we look at the rule for Character type specifier
:
R721 char-selector is length-selector
or ( LEN = type-param-value ,
KIND = scalar-int-constant-expr )
or ( type-param-value ,
[ KIND = ] scalar-int-constant-expr )
or ( KIND = scalar-int-constant-expr
[ , LEN =type-param-value ] )
R722 length-selector is ( [ LEN = ] type-param-value )
or * char-length [ , ]
R723 char-length is ( type-param-value )
or int-literal-constant
and:
R701 type-param-value is scalar-int-expr
or *
or :
so we can have * char-length
with char-length
as (*)
or 5
for instance, and they’re both forms of the obsolete rule. You also can’t tell from the text above, but in the standard * char-length
in R722 is in small case, indicating that it’s obsolescent.
Sometimes in old code you might see character foo(N)*N
. Is this actually covered under R721?
Looking further through the standard, there’s this bit for entity-decl
:
R803 entity-decl is object-name [ ( array-spec ) ]
[ lbracket coarray-spec rbracket ]
[ * char-length ] [ initialization ]
or function-name [ * char-length ]
with a comment:
C804 (R803) If the entity is not of type character, * char-length shall not appear.
Here * char-length
is not in small case and there’s no reference to R721
or char-selector
, and I think a really strict reading of B.3.8 means that it’s not obsolescent?! I guess because this might be the only way to have multiple character
variables with different lengths declared on the same line, similar to how (array-spec)
in R803 is how you can have different length arrays declared on the same line, but this feels surprising to me.
Looking at how different compilers treat the different forms:
subroutine foo(a, b, c)
implicit none
character*8 :: a
character*(*) :: b
character c(4)*4
end subroutine foo
Only gfortran and lfortran warn about any of them, and then they both only warn about the first form. I can see how potentially the last form isn’t technically obsolescent (but should be?), but my reading is definitely that the second form is and should be a warning. I could be missing some compiler flags for flang and ifx.
Any language lawyers able to give some insight into what’s going on here? I think the standard could be a bit more explicit either way!