Peter Klausler a3e9d3c2c7
[flang] Allow reference to earlier generic in later interface
Name resolutions defers all resolution and checking of specific procedures
in non-type-bound generic interfaces to the end of the specification part.
This prevents expression analysis of references to generic functions in
specification expressions in interfaces from resolving.

Example (now a new test case in modfile07.f90):
```
  module m12
    interface generic
      module procedure specific
    end interface
    interface
      module subroutine s(a1,a2)
        character(*) a1
        character(generic(a1)) a2   ! <--
      end
    end interface
   contains
    pure integer function specific(x)
      character(*), intent(in) :: x
      specific = len(x)
    end
  end
```

The solution is to partially resolve specific procedures as they are
defined for each generic, when they can be resolved, with the final
pass at the end of the specification part to finish up any forward
references and emit the necessary error messages.

Making this fix caused some issues in module file output, which have
all been resolved by making this simplifying change: generics are
now all emitted to module file specification parts as their own
group of declarations at the end of the specification part,
followed only by namelists and COMMON blocks.

Differential Revision: https://reviews.llvm.org/D157346
2023-08-08 12:19:53 -07:00

661 lines
10 KiB
Fortran

! RUN: %python %S/test_modfile.py %s %flang_fc1
! Check modfile generation for generic interfaces
module m1
interface foo
real function s1(x,y)
real, intent(in) :: x
logical, intent(in) :: y
end function
complex function s2(x,y)
complex, intent(in) :: x
logical, intent(in) :: y
end function
end interface
generic :: operator ( + ) => s1, s2
interface operator ( /= )
logical function f1(x, y)
real, intent(in) :: x
logical, intent(in) :: y
end function
end interface
interface
logical function f2(x, y)
complex, intent(in) :: x
logical, intent(in) :: y
end function
logical function f3(x, y)
integer, intent(in) :: x
logical, intent(in) :: y
end function
end interface
generic :: operator(.ne.) => f2
generic :: operator(<>) => f3
private :: operator( .ne. )
interface bar
procedure :: s1
procedure :: s2
procedure :: s3
procedure :: s4
end interface
interface operator( .bar.)
procedure :: s1
procedure :: s2
procedure :: s3
procedure :: s4
end interface
contains
logical function s3(x,y)
logical, intent(in) :: x,y
end function
integer function s4(x,y)
integer, intent(in) :: x,y
end function
end
!Expect: m1.mod
!module m1
! interface
! function s1(x,y)
! real(4),intent(in)::x
! logical(4),intent(in)::y
! real(4)::s1
! end
! end interface
! interface
! function s2(x,y)
! complex(4),intent(in)::x
! logical(4),intent(in)::y
! complex(4)::s2
! end
! end interface
! interface
! function f1(x,y)
! real(4),intent(in)::x
! logical(4),intent(in)::y
! logical(4)::f1
! end
! end interface
! interface
! function f2(x,y)
! complex(4),intent(in)::x
! logical(4),intent(in)::y
! logical(4)::f2
! end
! end interface
! interface
! function f3(x,y)
! integer(4),intent(in)::x
! logical(4),intent(in)::y
! logical(4)::f3
! end
! end interface
! interface foo
! procedure::s1
! procedure::s2
! end interface
! interface operator(+)
! procedure::s1
! procedure::s2
! end interface
! interface operator(/=)
! procedure::f1
! procedure::f2
! procedure::f3
! end interface
! private::operator(/=)
! interface bar
! procedure::s1
! procedure::s2
! procedure::s3
! procedure::s4
! end interface
! interface operator(.bar.)
! procedure::s1
! procedure::s2
! procedure::s3
! procedure::s4
! end interface
!contains
! function s3(x,y)
! logical(4),intent(in)::x
! logical(4),intent(in)::y
! logical(4)::s3
! end
! function s4(x,y)
! integer(4),intent(in)::x
! integer(4),intent(in)::y
! integer(4)::s4
! end
!end
module m1b
use m1
end
!Expect: m1b.mod
!module m1b
! use m1,only:foo
! use m1,only:s1
! use m1,only:s2
! use m1,only:operator(+)
! use m1,only:f1
! use m1,only:f2
! use m1,only:f3
! use m1,only:bar
! use m1,only:operator(.bar.)
! use m1,only:s3
! use m1,only:s4
!end
module m1c
use m1, only: myfoo => foo
use m1, only: operator(.bar.)
use m1, only: operator(.mybar.) => operator(.bar.)
use m1, only: operator(+)
end
!Expect: m1c.mod
!module m1c
! use m1,only:myfoo=>foo
! use m1,only:operator(.bar.)
! use m1,only:operator(.mybar.)=>operator(.bar.)
! use m1,only:operator(+)
!end
module m2
interface foo
procedure foo
end interface
contains
complex function foo()
foo = 1.0
end
end
!Expect: m2.mod
!module m2
! interface foo
! procedure::foo
! end interface
!contains
! function foo()
! complex(4)::foo
! end
!end
module m2b
type :: foo
real :: x
end type
interface foo
end interface
private :: bar
interface bar
end interface
end
!Expect: m2b.mod
!module m2b
! type::foo
! real(4)::x
! end type
! interface foo
! end interface
! interface bar
! end interface
! private::bar
!end
! Test interface nested inside another interface
module m3
interface g
subroutine s1(f)
interface
real function f(x)
interface
subroutine x()
end subroutine
end interface
end function
end interface
end subroutine
end interface
end
!Expect: m3.mod
!module m3
! interface
! subroutine s1(f)
! interface
! function f(x)
! interface
! subroutine x()
! end
! end interface
! real(4)::f
! end
! end interface
! end
! end interface
! interface g
! procedure::s1
! end interface
!end
module m4
interface foo
integer function foo()
end function
integer function f(x)
end function
end interface
end
subroutine s4
use m4
i = foo()
end
!Expect: m4.mod
!module m4
! interface
! function foo()
! integer(4)::foo
! end
! end interface
! interface
! function f(x)
! real(4)::x
! integer(4)::f
! end
! end interface
! interface foo
! procedure::foo
! procedure::f
! end interface
!end
! Compile contents of m4.mod and verify it gets the same thing again.
module m5
interface foo
procedure::foo
procedure::f
end interface
interface
function foo()
integer(4)::foo
end
end interface
interface
function f(x)
integer(4)::f
real(4)::x
end
end interface
end
!Expect: m5.mod
!module m5
! interface
! function foo()
! integer(4)::foo
! end
! end interface
! interface
! function f(x)
! real(4)::x
! integer(4)::f
! end
! end interface
! interface foo
! procedure::foo
! procedure::f
! end interface
!end
module m6a
interface operator(<)
logical function lt(x, y)
logical, intent(in) :: x, y
end function
end interface
end
!Expect: m6a.mod
!module m6a
! interface
! function lt(x,y)
! logical(4),intent(in)::x
! logical(4),intent(in)::y
! logical(4)::lt
! end
! end interface
! interface operator(<)
! procedure::lt
! end interface
!end
module m6b
use m6a, only: operator(.lt.)
end
!Expect: m6b.mod
!module m6b
! use m6a,only:operator(.lt.)
!end
module m7a
interface g_integer
module procedure s
end interface
private :: s
contains
subroutine s(x)
integer :: x
end
end
!Expect: m7a.mod
!module m7a
! private :: s
! interface g_integer
! procedure :: s
! end interface
!contains
! subroutine s(x)
! integer(4) :: x
! end
!end
module m7b
interface g_real
module procedure s
end interface
private :: s
contains
subroutine s(x)
real :: x
end subroutine
end
!Expect: m7b.mod
!module m7b
! private :: s
! interface g_real
! procedure :: s
! end interface
!contains
! subroutine s(x)
! real(4) :: x
! end
!end
module m7c
use m7a, only: g => g_integer
use m7b, only: g => g_real
interface g
module procedure s
end interface
private :: s
contains
subroutine s(x)
complex :: x
end subroutine
subroutine test()
real :: x
integer :: y
complex :: z
call g(x)
call g(y)
call g(z)
end
end
!Expect: m7c.mod
!module m7c
! use m7a, only: g => g_integer
! use m7b, only: g => g_real
! private :: s
! interface g
! procedure :: s
! end interface
!contains
! subroutine s(x)
! complex(4) :: x
! end
! subroutine test()
! end
!end
! Test m8 is like m7 but without renaming.
module m8a
interface g
module procedure s
end interface
private :: s
contains
subroutine s(x)
integer :: x
end
end
!Expect: m8a.mod
!module m8a
! private :: s
! interface g
! procedure :: s
! end interface
!contains
! subroutine s(x)
! integer(4) :: x
! end
!end
module m8b
interface g
module procedure s
end interface
private :: s
contains
subroutine s(x)
real :: x
end subroutine
end
!Expect: m8b.mod
!module m8b
! private :: s
! interface g
! procedure :: s
! end interface
!contains
! subroutine s(x)
! real(4) :: x
! end
!end
module m8c
use m8a
use m8b
interface g
module procedure s
end interface
private :: s
contains
subroutine s(x)
complex :: x
end subroutine
subroutine test()
real :: x
integer :: y
complex :: z
call g(x)
call g(y)
call g(z)
end
end
!Expect: m8c.mod
!module m8c
! use m8a, only: g
! use m8b, only: g
! private :: s
! interface g
! procedure :: s
! end interface
!contains
! subroutine s(x)
! complex(4) :: x
! end
! subroutine test()
! end
!end
! Merging a use-associated generic with a local generic
module m9a
interface g
module procedure s
end interface
private :: s
contains
subroutine s(x)
integer :: x
end
end
!Expect: m9a.mod
!module m9a
! private :: s
! interface g
! procedure :: s
! end interface
!contains
! subroutine s(x)
! integer(4) :: x
! end
!end
module m9b
use m9a
interface g
module procedure s
end interface
private :: s
contains
subroutine s(x)
real :: x
end
subroutine test()
call g(1)
call g(1.0)
end
end
!Expect: m9b.mod
!module m9b
! use m9a,only:g
! private::s
! interface g
! procedure::s
! end interface
!contains
! subroutine s(x)
! real(4)::x
! end
! subroutine test()
! end
!end
! Verify that equivalent names are used when generic operators are merged
module m10a
interface operator(.ne.)
end interface
end
!Expect: m10a.mod
!module m10a
! interface operator(.ne.)
! end interface
!end
module m10b
interface operator(<>)
end interface
end
!Expect: m10b.mod
!module m10b
! interface operator(<>)
! end interface
!end
module m10c
use m10a
use m10b
interface operator(/=)
end interface
end
!Expect: m10c.mod
!module m10c
! use m10a,only:operator(.ne.)
! use m10b,only:operator(.ne.)
! interface operator(.ne.)
! end interface
!end
module m10d
use m10a
use m10c
private :: operator(<>)
end
!Expect: m10d.mod
!module m10d
! use m10a,only:operator(.ne.)
! use m10c,only:operator(.ne.)
! interface operator(.ne.)
! end interface
! private::operator(.ne.)
!end
module m11a
contains
subroutine s1()
end
end
!Expect: m11a.mod
!module m11a
!contains
! subroutine s1()
! end
!end
module m11b
use m11a
interface g
module procedure s1
end interface
end
!Expect: m11b.mod
!module m11b
! use m11a,only:s1
! interface g
! procedure::s1
! end interface
!end
module m12
interface generic
module procedure specific
end interface
interface
module subroutine s(a1,a2)
character(*) a1
character(generic(a1)) a2
end
end interface
contains
pure integer function specific(x)
character(*), intent(in) :: x
specific = len(x)
end
end
!Expect: m12.mod
!module m12
! interface
! module subroutine s(a1,a2)
! character(*,1)::a1
! character(specific(a1),1)::a2
! end
! end interface
! interface generic
! procedure::specific
! end interface
!contains
! pure function specific(x)
! character(*,1),intent(in)::x
! integer(4)::specific
! end
!end