Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[F03] bogus error: Could not resolve generic type bound procedure #533 #849

Open
wants to merge 5 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Next Next commit
[F03] bogus error: Could not resolve generic type bound procedure #533
Earlier a tbp arg was added for few type bound procedure calls (with/without nopass clause).
This was inconsistent. While procedure matching, procedures with nopass clauses were
not considered.
This has been fixed.
Now a tbp arg will be added to all type bound procedure calls and this tbp arg will be
considered/discarded depending on the procedure (with/without nopass clause) being matched.

Change-Id: I73a1b0c15852b8d2302048d44db4aa26f1f0a9ac
  • Loading branch information
kiranktp committed Jan 8, 2020
commit 6aa81f765c5f438ae29596f0b474d018cde6122a
32 changes: 32 additions & 0 deletions test/f90_correct/inc/tbp.mk
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
# Copyright (c) 2019, Advanced Micro Devices, Inc. All rights reserved.
#
# Date of Modification: December 2019
#

########## Make rule to test type-bound procedures ########

fcheck.o check_mod.mod: $(SRC)/check_mod.f90
-$(FC) -c $(FFLAGS) $(SRC)/check_mod.f90 -o fcheck.o

tbp.o: $(SRC)/tbp.f90 check_mod.mod
@echo ------------------------------------ building test $@
-$(FC) -c $(FFLAGS) $(LDFLAGS) $(SRC)/tbp.f90 -o tbp.o

tbp: tbp.o fcheck.o
-$(FC) $(FFLAGS) $(LDFLAGS) tbp.o fcheck.o $(LIBS) -o tbp

tbp.run: tbp
@echo ------------------------------------ executing test tbp
tbp
-$(RM) test_m.mod

### TA Expected Targets ###

build: $(TEST)

.PHONY: run
run: $(TEST).run

verify: ;

### End of Expected Targets ###
7 changes: 7 additions & 0 deletions test/f90_correct/lit/tbp.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
# Copyright (c) 2019, Advanced Micro Devices, Inc. All rights reserved.
#
# Date of Modification: December 2019
#

# RUN: KEEP_FILES=%keep FLAGS=%flags TEST_SRC=%s MAKE_FILE_DIR=%S/.. bash %S/runmake | tee %t
# RUN: cat %t | FileCheck %S/runmake
91 changes: 91 additions & 0 deletions test/f90_correct/src/tbp.f90
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
! Copyright (c) 2019, Advanced Micro Devices, Inc. All rights reserved.
!
! Date of Modification: December 2019

module test_m
implicit none

type A_t
contains
! Case 1:
procedure ,nopass :: f_int
procedure :: f_real
generic :: f => f_int, f_real
! Case 2:
procedure :: f_int1
procedure ,nopass :: f_real1
generic :: f1 => f_int1, f_real1
! Case 3:
procedure ,nopass:: f_int2
procedure ,nopass :: f_real2
generic :: f2 => f_int2, f_real2
! Case 4:
procedure :: f_int3
procedure :: f_real3
generic :: f3 => f_int3, f_real3
endtype

contains
! Case 1:
integer function f_int( n ) result (RSLT)
integer :: n
RSLT = n - 1
end function f_int
integer function f_real( me, x ) result (RSLT)
class(A_t) :: me
real :: x
RSLT = x + 1
end function f_real

! Case 2:
integer function f_int1( me, n ) result (RSLT)
class(A_t) :: me
integer :: n
RSLT = n - 1
end function f_int1
integer function f_real1( x ) result (RSLT)
real :: x
RSLT = x + 1
end function f_real1

! Case 3:
integer function f_int2( n ) result (RSLT)
integer :: n
RSLT = n - 1
end function f_int2
integer function f_real2( x ) result (RSLT)
real :: x
RSLT = x + 1
end function f_real2

! Case 3:
integer function f_int3( me, n ) result (RSLT)
class(A_t) :: me
integer :: n
RSLT = n - 1
end function f_int3
integer function f_real3( me, x ) result (RSLT)
class(A_t) :: me
real :: x
RSLT = x + 1
end function f_real3
end module

program main
USE CHECK_MOD
use test_m
implicit none
type(A_t) :: A
logical results(4)
logical expect(4)

results = .false.
expect = .true.

results(1) = 9 .eq. A%f(10)
results(2) = 99 .eq. A%f1(100)
results(3) = 999 .eq. A%f2(1000)
results(4) = 9999 .eq. A%f3(10000)
Comment on lines +97 to +100
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you call the real ones also through the generic name?

Copy link
Author

@kiranktp kiranktp Feb 4, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I had fogotten to add a check for functions without any parameter. I will add them too.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you call the real ones also through the generic name?
Based on the type of the argument passed, the function call gets resolved. What do u mean by calling the real functions thru generic name.? We can always call the actual functions with their names.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was pointing out that you are calling only the integer ones. Call the real ones also in the test for completeness.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oops sorry, my bad. Sure I will add the check for real type as well.


call check(results,expect,4)
end
14 changes: 3 additions & 11 deletions tools/flang1/flang1exe/semant2.c
Original file line number Diff line number Diff line change
Expand Up @@ -768,16 +768,6 @@ semant2(int rednum, SST *top)
} else {
int dty = TBPLNKG(sptr);
itemp = ITEM_END;
if (generic_tbp_has_pass_and_nopass(dty, sptr)) {
int parent, sp;
e1 = (SST *)getitem(0, sizeof(SST));
sp = sym_of_ast(ast);
SST_SYMP(e1, sp);
SST_DTYPEP(e1, DTYPEG(sp));
mkident(e1);
mkexpr(e1);
itemp = mkitem(e1);
}
goto var_ref_common;
}
}
Expand Down Expand Up @@ -966,7 +956,9 @@ semant2(int rednum, SST *top)
mem2 = get_specific_member(TBPLNKG(sptr), VTABLEG(mem));
argno = get_tbp_argno(BINDG(mem2), TBPLNKG(sptr));
if (!argno && NOPASSG(mem2)) {
goto var_ref_common; /* assume NOPASS tbp */
// One tbp argument will be added to a type bound procedure call
// with NOPASS clause.
argno = 1;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is the same change not required in the else portion below?

Can you add a test for operator also?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This check is for generic type bound procedures only, so we do not need this for else part.
I tried to come up with a test case for operator to see if this fix has any side effects but I couldn't.
I also checked if any test case from flang testing infrastructure matches this scenario. none of the tests has the case which enters this piece of code with operator.
This piece of code is being used only by type bound procedures.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it is better to restrict this change only to generic type bound procedures. I will do this change and push it for review again.

}
} else {
argno = get_tbp_argno(sptr, DTYPEG(pass_sym_of_ast(ast)));
Expand Down
61 changes: 61 additions & 0 deletions tools/flang1/flang1exe/semfunc.c
Original file line number Diff line number Diff line change
Expand Up @@ -654,6 +654,36 @@ is_ptr_arg(SST *sst_actual)
return sptr > NOSYM && POINTERG(sptr);
}

// AOCC Begin
// Add a tbp arg when there is a call to type bound procedures
static ITEM*
add_tbp_arg (SST *stktop, ITEM *itemp)
{
ITEM *itemp2;
SST *e1em;
int sp;
int ast = SST_ASTG(stktop);
e1em = (SST *)getitem(0, sizeof(SST));
sp = sym_of_ast(ast);
SST_SYMP(e1em, sp);
SST_DTYPEP(e1em, DTYPEG(sp));
mkident(e1em);
mkexpr(e1em);
itemp2 = (ITEM *)getitem(0, sizeof(ITEM));
itemp2->t.stkp = e1em;
itemp2->next = ITEM_END;

//tbp arg will be the first argument
if (itemp == ITEM_END) {
itemp = itemp2;
} else {
itemp2->next = itemp;
itemp = itemp2;
}
return itemp;
} // add_tbp_arg
// AOCC End

/* Non-pointer passed to a pointer dummy: geneerate a pointer temp, associate
* the temp with the actual arg, and pass the temp.
*/
Expand Down Expand Up @@ -850,6 +880,14 @@ func_call2(SST *stktop, ITEM *list, int flag)
dtype = DTY(dtype + 1);
if (STYPEG(BINDG(callee)) == ST_USERGENERIC) {
int mem;
int imp, mem1;
// For type bound procedures with no "nopass" clause, tbp arg
// has already been added to the list. Need to do the same for type bound
// procedures with "nopass" clause as well.
sptr1 = BINDG(callee);
imp = get_implementation(TBPLNKG(sptr1), sptr1, 0, &mem1);
if (imp && NOPASSG(mem1))
list = add_tbp_arg(stktop, list);
func_sptr = generic_tbp_func(BINDG(callee), stktop, list);
if (func_sptr) {
if (get_implementation(dtype, func_sptr, 0, &mem) == 0) {
Expand All @@ -867,6 +905,13 @@ func_call2(SST *stktop, ITEM *list, int flag)
} else {
SST_ASTP(stktop, replace_memsym_of_ast(SST_ASTG(stktop), mem));
callee = mem;
// For the type bound procedures with nopass clause,
// tbg arg should be removed now.
// Procedure has already been resolved.
// First argument is tbp arg.
if (NOPASSG(mem)) {
list = list->next;
}
}
}
}
Expand Down Expand Up @@ -3262,6 +3307,13 @@ subr_call2(SST *stktop, ITEM *list, int flag)
}
if (stype == ST_USERGENERIC && check_generic) {
if (CLASSG(sptr)) {
int imp, mem;
imp = get_implementation(TBPLNKG(sptr), sptr, 0, &mem);
// For type bound procedures with no "nopass" clause, tbp arg
// has already been added to the list. Need to do the same for type bound
// procedures with "nopass" clause as well.
if (imp && NOPASSG(mem))
list = add_tbp_arg(stktop, list);
sptr = generic_tbp_call(sptr, stktop, list, 0);
goto do_call;
}
Expand Down Expand Up @@ -3427,6 +3479,15 @@ subr_call2(SST *stktop, ITEM *list, int flag)
sptr1 = 0;
break;
}
// For the type bound procedures with nopass clause,
// tbg arg should be removed now.
// Procedure has already been resolved.
// First argument is tbp arg.
if (NOPASSG(mem)) {
list = list->next;
count_actuals(list);
count = carg.nent;
}
ast = replace_memsym_of_ast(ast, mem);
SST_ASTP(stktop, ast);
sptr = BINDG(mem);
Expand Down
14 changes: 12 additions & 2 deletions tools/flang1/flang1exe/semgnr.c
Original file line number Diff line number Diff line change
Expand Up @@ -313,6 +313,8 @@ find_best_generic(int gnr, ITEM *list, int arg_cnt, int try_device,
int dscptr;
int paramct, curr_paramct;
SPTR found_sptrgen, func_sptrgen;
ITEM *list_bkp = list;
int arg_cnt_bkp = arg_cnt;

/* find the generic's max nbr of formal args and use it to compute
* the size of the arg distatnce data item.
Expand Down Expand Up @@ -382,6 +384,9 @@ find_best_generic(int gnr, ITEM *list, int arg_cnt, int try_device,
for (gndsc = GNDSCG(sptrgen); gndsc; gndsc = SYMI_NEXT(gndsc)) {
func = SYMI_SPTR(gndsc);
func_sptrgen = sptrgen;
// Restore the argument list and argument count
list = list_bkp;
arg_cnt = arg_cnt_bkp;
if (IS_TBP(func)) {
/* For generic type bound procedures, use the implementation
* of the generic bind name for the argument comparison.
Expand All @@ -399,8 +404,13 @@ find_best_generic(int gnr, ITEM *list, int arg_cnt, int try_device,
if (!func)
continue;
mem = get_generic_member(dty, bind);
if (NOPASSG(mem) && generic_tbp_has_pass_and_nopass(dty, BINDG(mem)))
continue;
if (NOPASSG(mem)) {
// skip the tbp arg which has been added while processing the call
// before matching the procedure.
// type bound procedures with nopass clause will not have tbp argument.
list = list->next;
arg_cnt--;
}
if (mem && PRIVATEG(mem) && SCOPEG(stb.curr_scope) != SCOPEG(mem))
continue;
} else
Expand Down