diff options
Diffstat (limited to 'gcc/fortran/openmp.cc')
-rw-r--r-- | gcc/fortran/openmp.cc | 53 |
1 files changed, 41 insertions, 12 deletions
diff --git a/gcc/fortran/openmp.cc b/gcc/fortran/openmp.cc index 79c0f1b..e00044d 100644 --- a/gcc/fortran/openmp.cc +++ b/gcc/fortran/openmp.cc @@ -408,7 +408,8 @@ gfc_match_omp_variable_list (const char *str, gfc_omp_namelist **list, bool allow_sections = false, bool allow_derived = false, bool *has_all_memory = NULL, - bool reject_common_vars = false) + bool reject_common_vars = false, + bool reverse_order = false) { gfc_omp_namelist *head, *tail, *p; locus old_loc, cur_loc; @@ -492,15 +493,20 @@ gfc_match_omp_variable_list (const char *str, gfc_omp_namelist **list, p = gfc_get_omp_namelist (); if (head == NULL) head = tail = p; + else if (reverse_order) + { + p->next = head; + head = p; + } else { tail->next = p; tail = tail->next; } - tail->sym = sym; - tail->expr = expr; - tail->where = gfc_get_location_range (NULL, 0, &cur_loc, 1, - &gfc_current_locus); + p->sym = sym; + p->expr = expr; + p->where = gfc_get_location_range (NULL, 0, &cur_loc, 1, + &gfc_current_locus); if (reject_common_vars && sym->attr.in_common) { gcc_assert (allow_common); @@ -540,13 +546,18 @@ gfc_match_omp_variable_list (const char *str, gfc_omp_namelist **list, p = gfc_get_omp_namelist (); if (head == NULL) head = tail = p; + else if (reverse_order) + { + p->next = head; + head = p; + } else { tail->next = p; tail = tail->next; } - tail->sym = sym; - tail->where = cur_loc; + p->sym = sym; + p->where = cur_loc; } next_item: @@ -1128,6 +1139,7 @@ enum omp_mask2 OMP_CLAUSE_USE, /* OpenMP 5.1. */ OMP_CLAUSE_NOVARIANTS, /* OpenMP 5.1 */ OMP_CLAUSE_NOCONTEXT, /* OpenMP 5.1 */ + OMP_CLAUSE_INTEROP, /* OpenMP 5.1 */ /* This must come last. */ OMP_MASK2_LAST }; @@ -3255,6 +3267,21 @@ gfc_match_omp_clauses (gfc_omp_clauses **cp, const omp_mask mask, continue; goto error; } + if ((mask & OMP_CLAUSE_INTEROP) + && (m = gfc_match_dupl_check (!c->lists[OMP_LIST_INTEROP], + "interop", true)) != MATCH_NO) + { + /* Note: the interop objects are saved in reverse order to match + the order in C/C++. */ + if (m == MATCH_YES + && (gfc_match_omp_variable_list ("", + &c->lists[OMP_LIST_INTEROP], + false, NULL, NULL, false, + false, NULL, false, true) + == MATCH_YES)) + continue; + goto error; + } if ((mask & OMP_CLAUSE_IS_DEVICE_PTR) && gfc_match_omp_variable_list ("is_device_ptr (", @@ -5019,7 +5046,7 @@ cleanup: #define OMP_DISPATCH_CLAUSES \ (omp_mask (OMP_CLAUSE_DEVICE) | OMP_CLAUSE_DEPEND | OMP_CLAUSE_NOVARIANTS \ | OMP_CLAUSE_NOCONTEXT | OMP_CLAUSE_IS_DEVICE_PTR | OMP_CLAUSE_NOWAIT \ - | OMP_CLAUSE_HAS_DEVICE_ADDR) + | OMP_CLAUSE_HAS_DEVICE_ADDR | OMP_CLAUSE_INTEROP) static match @@ -8128,7 +8155,7 @@ resolve_omp_clauses (gfc_code *code, gfc_omp_clauses *omp_clauses, "DEVICE_RESIDENT", "LINK", "USE_DEVICE", "CACHE", "IS_DEVICE_PTR", "USE_DEVICE_PTR", "USE_DEVICE_ADDR", "NONTEMPORAL", "ALLOCATE", "HAS_DEVICE_ADDR", "ENTER", - "USES_ALLOCATORS", "INIT", "USE", "DESTROY", "ADJUST_ARGS" }; + "USES_ALLOCATORS", "INIT", "USE", "DESTROY", "INTEROP", "ADJUST_ARGS" }; STATIC_ASSERT (ARRAY_SIZE (clause_names) == OMP_LIST_NUM); if (omp_clauses == NULL) @@ -8455,6 +8482,7 @@ resolve_omp_clauses (gfc_code *code, gfc_omp_clauses *omp_clauses, && list != OMP_LIST_DEPEND && list != OMP_LIST_FROM && list != OMP_LIST_TO + && list != OMP_LIST_INTEROP && (list != OMP_LIST_REDUCTION || !openacc) && list != OMP_LIST_ALLOCATE) for (n = omp_clauses->lists[list]; n; n = n->next) @@ -8553,8 +8581,8 @@ resolve_omp_clauses (gfc_code *code, gfc_omp_clauses *omp_clauses, break; } } - if (code && code->op == EXEC_OMP_INTEROP) - for (list = OMP_LIST_INIT; list <= OMP_LIST_DESTROY; list++) + if (code && (code->op == EXEC_OMP_INTEROP || code->op == EXEC_OMP_DISPATCH)) + for (list = OMP_LIST_INIT; list <= OMP_LIST_INTEROP; list++) for (n = omp_clauses->lists[list]; n; n = n->next) { if (n->sym->ts.type != BT_INTEGER @@ -8564,7 +8592,8 @@ resolve_omp_clauses (gfc_code *code, gfc_omp_clauses *omp_clauses, gfc_error ("%qs at %L in %qs clause must be a scalar integer " "variable of %<omp_interop_kind%> kind", n->sym->name, &n->where, clause_names[list]); - if (list != OMP_LIST_USE && n->sym->attr.intent == INTENT_IN) + if (list != OMP_LIST_USE && list != OMP_LIST_INTEROP + && n->sym->attr.intent == INTENT_IN) gfc_error ("%qs at %L in %qs clause must be definable", n->sym->name, &n->where, clause_names[list]); } |