Skip to content

Commit 3bcab6f

Browse files
authoredMar 21, 2025
[flang][OpenMP][Semantics] improve semantic checks for array sections (#132230)
I'm not sure why strides were not allowed in array sections: the stride is explicitly allowed by the standard from the first version where array sections were introduced. The limitation is that the stride must not be negative. Here I have added the check for a negative stride and updated the test for a zero length section to take account of the stride.
1 parent 44261da commit 3bcab6f

File tree

2 files changed

+35
-18
lines changed

2 files changed

+35
-18
lines changed
 

‎flang/lib/Semantics/check-omp-structure.cpp

+26-16
Original file line numberDiff line numberDiff line change
@@ -5363,27 +5363,37 @@ void OmpStructureChecker::CheckArraySection(
53635363
if (const auto *triplet{
53645364
std::get_if<parser::SubscriptTriplet>(&subscript.u)}) {
53655365
if (std::get<0>(triplet->t) && std::get<1>(triplet->t)) {
5366+
std::optional<int64_t> strideVal{std::nullopt};
5367+
if (const auto &strideExpr = std::get<2>(triplet->t)) {
5368+
// OpenMP 6.0 Section 5.2.5: Array Sections
5369+
// Restrictions: if a stride expression is specified it must be
5370+
// positive. A stride of 0 doesn't make sense.
5371+
strideVal = GetIntValue(strideExpr);
5372+
if (strideVal && *strideVal < 1) {
5373+
context_.Say(GetContext().clauseSource,
5374+
"'%s' in %s clause must have a positive stride"_err_en_US,
5375+
name.ToString(),
5376+
parser::ToUpperCaseLetters(getClauseName(clause).str()));
5377+
}
5378+
}
53665379
const auto &lower{std::get<0>(triplet->t)};
53675380
const auto &upper{std::get<1>(triplet->t)};
53685381
if (lower && upper) {
53695382
const auto lval{GetIntValue(lower)};
53705383
const auto uval{GetIntValue(upper)};
5371-
if (lval && uval && *uval < *lval) {
5372-
context_.Say(GetContext().clauseSource,
5373-
"'%s' in %s clause"
5374-
" is a zero size array section"_err_en_US,
5375-
name.ToString(),
5376-
parser::ToUpperCaseLetters(getClauseName(clause).str()));
5377-
break;
5378-
} else if (std::get<2>(triplet->t)) {
5379-
const auto &strideExpr{std::get<2>(triplet->t)};
5380-
if (strideExpr) {
5381-
if (clause == llvm::omp::Clause::OMPC_depend) {
5382-
context_.Say(GetContext().clauseSource,
5383-
"Stride should not be specified for array section in "
5384-
"DEPEND "
5385-
"clause"_err_en_US);
5386-
}
5384+
if (lval && uval) {
5385+
int64_t sectionLen = *uval - *lval;
5386+
if (strideVal) {
5387+
sectionLen = sectionLen / *strideVal;
5388+
}
5389+
5390+
if (sectionLen < 1) {
5391+
context_.Say(GetContext().clauseSource,
5392+
"'%s' in %s clause"
5393+
" is a zero size array section"_err_en_US,
5394+
name.ToString(),
5395+
parser::ToUpperCaseLetters(getClauseName(clause).str()));
5396+
break;
53875397
}
53885398
}
53895399
}

‎flang/test/Semantics/OpenMP/depend01.f90

+9-2
Original file line numberDiff line numberDiff line change
@@ -17,8 +17,15 @@ program omp_depend
1717
a(2:1) = b(2, 2)
1818
!$omp end task
1919

20-
!ERROR: Stride should not be specified for array section in DEPEND clause
21-
!$omp task shared(x) depend(in: a(5:10:1))
20+
!ERROR: 'a' in DEPEND clause must have a positive stride
21+
!ERROR: 'b' in DEPEND clause must have a positive stride
22+
!ERROR: 'b' in DEPEND clause is a zero size array section
23+
!$omp task shared(x) depend(in: a(10:5:-1)) depend(in: b(5:10:-1))
24+
print *, a(5:10), b
25+
!$omp end task
26+
27+
!ERROR: 'a' in DEPEND clause is a zero size array section
28+
!$omp task shared(x) depend(in: a(1:5:10))
2229
print *, a(5:10), b
2330
!$omp end task
2431

0 commit comments

Comments
 (0)