mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-17 14:26:49 +00:00

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
661 lines
10 KiB
Fortran
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
|