OSDN Git Service

2011-02-16 Janus Weil <janus@gcc.gnu.org>
authorjanus <janus@138bc75d-0d04-0410-961f-82ee72b054a4>
Wed, 16 Feb 2011 20:51:56 +0000 (20:51 +0000)
committerjanus <janus@138bc75d-0d04-0410-961f-82ee72b054a4>
Wed, 16 Feb 2011 20:51:56 +0000 (20:51 +0000)
PR fortran/47745
* class.c (gfc_build_class_symbol): Set 'class_ok' attribute.
* decl.c (build_sym,attr_decl1): Move setting of 'class_ok' into
'gfc_build_class_symbol'.
(gfc_match_decl_type_spec): Reject unlimited polymorphism.
* interface.c (matching_typebound_op): Check for 'class_ok' attribute.
* match.c (select_type_set_tmp): Move setting of 'class_ok' into
'gfc_build_class_symbol'.
* primary.c (gfc_variable_attr): Check for 'class_ok' attribute.

2011-02-16  Janus Weil  <janus@gcc.gnu.org>

PR fortran/47745
* gfortran.dg/class_39.f03: New.

git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@170223 138bc75d-0d04-0410-961f-82ee72b054a4

gcc/fortran/ChangeLog
gcc/fortran/class.c
gcc/fortran/decl.c
gcc/fortran/interface.c
gcc/fortran/match.c
gcc/fortran/primary.c
gcc/testsuite/ChangeLog
gcc/testsuite/gfortran.dg/class_39.f03 [new file with mode: 0644]

index 346bb9e..340df01 100644 (file)
@@ -1,3 +1,15 @@
+2011-02-16  Janus Weil  <janus@gcc.gnu.org>
+
+       PR fortran/47745
+       * class.c (gfc_build_class_symbol): Set 'class_ok' attribute.
+       * decl.c (build_sym,attr_decl1): Move setting of 'class_ok' into
+       'gfc_build_class_symbol'.
+       (gfc_match_decl_type_spec): Reject unlimited polymorphism.
+       * interface.c (matching_typebound_op): Check for 'class_ok' attribute.
+       * match.c (select_type_set_tmp): Move setting of 'class_ok' into
+       'gfc_build_class_symbol'.
+       * primary.c (gfc_variable_attr): Check for 'class_ok' attribute.
+
 2011-02-15  Steven G. Kargl  <kargl@gcc.gnu.org>
 
        PR fortran/47633
index 67f19f7..85da3cb 100644 (file)
@@ -183,6 +183,16 @@ gfc_build_class_symbol (gfc_typespec *ts, symbol_attribute *attr,
   gfc_symbol *fclass;
   gfc_symbol *vtab;
   gfc_component *c;
+  
+  if (attr->class_ok)
+    /* Class container has already been built.  */
+    return SUCCESS;
+
+  attr->class_ok = attr->dummy || attr->pointer  || attr->allocatable;
+  
+  if (!attr->class_ok)
+    /* We can not build the class container yet.  */
+    return SUCCESS;
 
   if (*as)
     {
index 9712ea2..8b5f92b 100644 (file)
@@ -1177,9 +1177,7 @@ build_sym (const char *name, gfc_charlen *cl, bool cl_deferred,
 
   sym->attr.implied_index = 0;
 
-  if (sym->ts.type == BT_CLASS
-      && (sym->attr.class_ok = sym->attr.dummy || sym->attr.pointer
-                              || sym->attr.allocatable))
+  if (sym->ts.type == BT_CLASS)
     return gfc_build_class_symbol (&sym->ts, &sym->attr, &sym->as, false);
 
   return SUCCESS;
@@ -2613,6 +2611,16 @@ gfc_match_decl_type_spec (gfc_typespec *ts, int implicit_flag)
     ts->type = BT_DERIVED;
   else
     {
+      /* Match CLASS declarations.  */
+      m = gfc_match (" class ( * )");
+      if (m == MATCH_ERROR)
+       return MATCH_ERROR;
+      else if (m == MATCH_YES)
+       {
+         gfc_fatal_error ("Unlimited polymorphism at %C not yet supported");
+         return MATCH_ERROR;
+       }
+
       m = gfc_match (" class ( %n )", name);
       if (m != MATCH_YES)
        return m;
@@ -6045,9 +6053,7 @@ attr_decl1 (void)
        }
     }
     
-  if (sym->ts.type == BT_CLASS && !sym->attr.class_ok
-      && (sym->attr.class_ok = sym->attr.class_ok || current_attr.allocatable
-                              || current_attr.pointer)
+  if (sym->ts.type == BT_CLASS
       && gfc_build_class_symbol (&sym->ts, &sym->attr, &sym->as, false) == FAILURE)
     {
       m = MATCH_ERROR;
index 071eed9..b0b74c1 100644 (file)
@@ -2924,7 +2924,11 @@ matching_typebound_op (gfc_expr** tb_base,
        gfc_try result;
 
        if (base->expr->ts.type == BT_CLASS)
-         derived = CLASS_DATA (base->expr)->ts.u.derived;
+         {
+           if (!gfc_expr_attr (base->expr).class_ok)
+             continue;
+           derived = CLASS_DATA (base->expr)->ts.u.derived;
+         }
        else
          derived = base->expr->ts.u.derived;
 
index 01b88ff..d2d9f5f 100644 (file)
@@ -4536,11 +4536,8 @@ select_type_set_tmp (gfc_typespec *ts)
   gfc_add_pointer (&tmp->n.sym->attr, NULL);
   gfc_add_flavor (&tmp->n.sym->attr, FL_VARIABLE, name, NULL);
   if (ts->type == BT_CLASS)
-    {
-      gfc_build_class_symbol (&tmp->n.sym->ts, &tmp->n.sym->attr,
-                             &tmp->n.sym->as, false);
-      tmp->n.sym->attr.class_ok = 1;
-    }
+    gfc_build_class_symbol (&tmp->n.sym->ts, &tmp->n.sym->attr,
+                           &tmp->n.sym->as, false);
   tmp->n.sym->attr.select_type_temporary = 1;
 
   /* Add an association for it, so the rest of the parser knows it is
index b673e0b..c8e2bb6 100644 (file)
@@ -2033,7 +2033,7 @@ gfc_variable_attr (gfc_expr *expr, gfc_typespec *ts)
   sym = expr->symtree->n.sym;
   attr = sym->attr;
 
-  if (sym->ts.type == BT_CLASS)
+  if (sym->ts.type == BT_CLASS && sym->attr.class_ok)
     {
       dimension = CLASS_DATA (sym)->attr.dimension;
       pointer = CLASS_DATA (sym)->attr.class_pointer;
index e11bdf3..19488e8 100644 (file)
@@ -1,3 +1,8 @@
+2011-02-16  Janus Weil  <janus@gcc.gnu.org>
+
+       PR fortran/47745
+       * gfortran.dg/class_39.f03: New.
+
 2011-02-16  Dodji Seketeli  <dodji@redhat.com>
 
        PR c++/47326
diff --git a/gcc/testsuite/gfortran.dg/class_39.f03 b/gcc/testsuite/gfortran.dg/class_39.f03
new file mode 100644 (file)
index 0000000..bc8039f
--- /dev/null
@@ -0,0 +1,13 @@
+! { dg-do compile }
+!
+! PR 47745: [OOP] Segfault with CLASS(*) and derived type dummy arguments
+!
+! Contributed by Rodney Polkinghorne <thisrod@gmail.com>
+
+  type, abstract :: T 
+  end type T
+contains
+  class(T) function add()  ! { dg-error "must be dummy, allocatable or pointer" }
+    add = 1  ! { dg-error "Variable must not be polymorphic in assignment" }
+  end function
+end