1 //===-- lib/Semantics/data-to-inits.cpp -----------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // DATA statement object/value checking and conversion to static
11 // - Applies specific checks to each scalar element initialization with a
12 // constant value or pointer target with class DataInitializationCompiler;
13 // - Collects the elemental initializations for each symbol and converts them
14 // into a single init() expression with member function
15 // DataChecker::ConstructInitializer().
17 #include "data-to-inits.h"
18 #include "pointer-assignment.h"
19 #include "flang/Evaluate/fold-designator.h"
20 #include "flang/Evaluate/tools.h"
21 #include "flang/Semantics/tools.h"
23 // The job of generating explicit static initializers for objects that don't
24 // have them in order to implement default component initialization is now being
25 // done in lowering, so don't do it here in semantics; but the code remains here
26 // in case we change our minds.
27 static constexpr bool makeDefaultInitializationExplicit
{false};
29 // Whether to delete the original "init()" initializers from storage-associated
30 // objects and pointers.
31 static constexpr bool removeOriginalInits
{false};
33 // Impose a hard limit that's more than large enough for real applications but
34 // small enough to cause artificial stress tests to fail reasonably instead of
35 // crashing the compiler with a memory allocation failure.
36 static constexpr auto maxDataInitBytes
{std::size_t{1000000000}}; // 1GiB
38 namespace Fortran::semantics
{
40 // Steps through a list of values in a DATA statement set; implements
42 template <typename DSV
= parser::DataStmtValue
> class ValueListIterator
{
44 ValueListIterator(SemanticsContext
&context
, const std::list
<DSV
> &list
)
45 : context_
{context
}, end_
{list
.end()}, at_
{list
.begin()} {
48 bool hasFatalError() const { return hasFatalError_
; }
49 bool IsAtEnd() const { return at_
== end_
; }
50 const SomeExpr
*operator*() const { return GetExpr(context_
, GetConstant()); }
51 std::optional
<parser::CharBlock
> LocateSource() const {
52 if (!hasFatalError_
) {
53 return GetConstant().source
;
57 ValueListIterator
&operator++() {
58 if (repetitionsRemaining_
> 0) {
59 --repetitionsRemaining_
;
60 } else if (at_
!= end_
) {
68 using listIterator
= typename
std::list
<DSV
>::const_iterator
;
69 void SetRepetitionCount();
70 const parser::DataStmtValue
&GetValue() const {
71 return DEREF(common::Unwrap
<const parser::DataStmtValue
>(*at_
));
73 const parser::DataStmtConstant
&GetConstant() const {
74 return std::get
<parser::DataStmtConstant
>(GetValue().t
);
77 SemanticsContext
&context_
;
78 listIterator end_
, at_
;
79 ConstantSubscript repetitionsRemaining_
{0};
80 bool hasFatalError_
{false};
83 template <typename DSV
> void ValueListIterator
<DSV
>::SetRepetitionCount() {
84 for (; at_
!= end_
; ++at_
) {
85 auto repetitions
{GetValue().repetitions
};
86 if (repetitions
< 0) {
87 hasFatalError_
= true;
88 } else if (repetitions
> 0) {
89 repetitionsRemaining_
= repetitions
- 1;
93 repetitionsRemaining_
= 0;
96 // Collects all of the elemental initializations from DATA statements
97 // into a single image for each symbol that appears in any DATA.
98 // Expands the implied DO loops and array references.
99 // Applies checks that validate each distinct elemental initialization
100 // of the variables in a data-stmt-set, as well as those that apply
101 // to the corresponding values being used to initialize each element.
102 template <typename DSV
= parser::DataStmtValue
>
103 class DataInitializationCompiler
{
105 DataInitializationCompiler(DataInitializations
&inits
,
106 evaluate::ExpressionAnalyzer
&a
, const std::list
<DSV
> &list
)
107 : inits_
{inits
}, exprAnalyzer_
{a
}, values_
{a
.context(), list
} {}
108 const DataInitializations
&inits() const { return inits_
; }
109 bool HasSurplusValues() const { return !values_
.IsAtEnd(); }
110 bool Scan(const parser::DataStmtObject
&);
111 // Initializes all elements of whole variable or component
112 bool Scan(const Symbol
&);
115 bool Scan(const parser::Variable
&);
116 bool Scan(const parser::Designator
&);
117 bool Scan(const parser::DataImpliedDo
&);
118 bool Scan(const parser::DataIDoObject
&);
120 // Initializes all elements of a designator, which can be an array or section.
121 bool InitDesignator(const SomeExpr
&, const Scope
&);
122 // Initializes a single scalar object.
123 bool InitElement(const evaluate::OffsetSymbol
&, const SomeExpr
&designator
,
125 // If the returned flag is true, emit a warning about CHARACTER misusage.
126 std::optional
<std::pair
<SomeExpr
, bool>> ConvertElement(
127 const SomeExpr
&, const evaluate::DynamicType
&);
129 DataInitializations
&inits_
;
130 evaluate::ExpressionAnalyzer
&exprAnalyzer_
;
131 ValueListIterator
<DSV
> values_
;
134 template <typename DSV
>
135 bool DataInitializationCompiler
<DSV
>::Scan(
136 const parser::DataStmtObject
&object
) {
137 return common::visit(
139 [&](const common::Indirection
<parser::Variable
> &var
) {
140 return Scan(var
.value());
142 [&](const parser::DataImpliedDo
&ido
) { return Scan(ido
); },
147 template <typename DSV
>
148 bool DataInitializationCompiler
<DSV
>::Scan(const parser::Variable
&var
) {
149 if (const auto *expr
{GetExpr(exprAnalyzer_
.context(), var
)}) {
150 parser::CharBlock at
{var
.GetSource()};
151 exprAnalyzer_
.GetFoldingContext().messages().SetLocation(at
);
152 if (InitDesignator(*expr
, exprAnalyzer_
.context().FindScope(at
))) {
159 template <typename DSV
>
160 bool DataInitializationCompiler
<DSV
>::Scan(
161 const parser::Designator
&designator
) {
163 { // The out-of-range subscript errors from the designator folder are a
164 // more specific than the default ones from expression semantics, so
165 // disable those to avoid piling on.
166 auto restorer
{exprAnalyzer_
.GetContextualMessages().DiscardMessages()};
167 expr
= exprAnalyzer_
.Analyze(designator
);
170 parser::CharBlock at
{parser::FindSourceLocation(designator
)};
171 exprAnalyzer_
.GetFoldingContext().messages().SetLocation(at
);
172 if (InitDesignator(*expr
, exprAnalyzer_
.context().FindScope(at
))) {
179 template <typename DSV
>
180 bool DataInitializationCompiler
<DSV
>::Scan(const parser::DataImpliedDo
&ido
) {
181 const auto &bounds
{std::get
<parser::DataImpliedDo::Bounds
>(ido
.t
)};
182 auto name
{bounds
.name
.thing
.thing
};
183 const auto *lowerExpr
{
184 GetExpr(exprAnalyzer_
.context(), bounds
.lower
.thing
.thing
)};
185 const auto *upperExpr
{
186 GetExpr(exprAnalyzer_
.context(), bounds
.upper
.thing
.thing
)};
187 const auto *stepExpr
{bounds
.step
188 ? GetExpr(exprAnalyzer_
.context(), bounds
.step
->thing
.thing
)
190 if (lowerExpr
&& upperExpr
) {
191 // Fold the bounds expressions (again) in case any of them depend
192 // on outer implied DO loops.
193 evaluate::FoldingContext
&context
{exprAnalyzer_
.GetFoldingContext()};
194 std::int64_t stepVal
{1};
196 auto foldedStep
{evaluate::Fold(context
, SomeExpr
{*stepExpr
})};
197 stepVal
= ToInt64(foldedStep
).value_or(1);
199 exprAnalyzer_
.Say(name
.source
,
200 "DATA statement implied DO loop has a step value of zero"_err_en_US
);
204 auto foldedLower
{evaluate::Fold(context
, SomeExpr
{*lowerExpr
})};
205 auto lower
{ToInt64(foldedLower
)};
206 auto foldedUpper
{evaluate::Fold(context
, SomeExpr
{*upperExpr
})};
207 auto upper
{ToInt64(foldedUpper
)};
208 if (lower
&& upper
) {
209 int kind
{evaluate::ResultType
<evaluate::ImpliedDoIndex
>::kind
};
210 if (const auto dynamicType
{evaluate::DynamicType::From(*name
.symbol
)}) {
211 if (dynamicType
->category() == TypeCategory::Integer
) {
212 kind
= dynamicType
->kind();
215 if (exprAnalyzer_
.AddImpliedDo(name
.source
, kind
)) {
216 auto &value
{context
.StartImpliedDo(name
.source
, *lower
)};
218 for (auto n
{(*upper
- value
+ stepVal
) / stepVal
}; n
> 0;
219 --n
, value
+= stepVal
) {
220 for (const auto &object
:
221 std::get
<std::list
<parser::DataIDoObject
>>(ido
.t
)) {
228 context
.EndImpliedDo(name
.source
);
229 exprAnalyzer_
.RemoveImpliedDo(name
.source
);
237 template <typename DSV
>
238 bool DataInitializationCompiler
<DSV
>::Scan(
239 const parser::DataIDoObject
&object
) {
240 return common::visit(
242 [&](const parser::Scalar
<common::Indirection
<parser::Designator
>>
243 &var
) { return Scan(var
.thing
.value()); },
244 [&](const common::Indirection
<parser::DataImpliedDo
> &ido
) {
245 return Scan(ido
.value());
251 template <typename DSV
>
252 bool DataInitializationCompiler
<DSV
>::Scan(const Symbol
&symbol
) {
253 auto designator
{exprAnalyzer_
.Designate(evaluate::DataRef
{symbol
})};
254 CHECK(designator
.has_value());
255 return InitDesignator(*designator
, symbol
.owner());
258 template <typename DSV
>
259 bool DataInitializationCompiler
<DSV
>::InitDesignator(
260 const SomeExpr
&designator
, const Scope
&scope
) {
261 evaluate::FoldingContext
&context
{exprAnalyzer_
.GetFoldingContext()};
262 evaluate::DesignatorFolder folder
{context
};
263 while (auto offsetSymbol
{folder
.FoldDesignator(designator
)}) {
264 if (folder
.isOutOfRange()) {
265 if (auto bad
{evaluate::OffsetToDesignator(context
, *offsetSymbol
)}) {
266 exprAnalyzer_
.context().Say(
267 "DATA statement designator '%s' is out of range"_err_en_US
,
270 exprAnalyzer_
.context().Say(
271 "DATA statement designator '%s' is out of range"_err_en_US
,
272 designator
.AsFortran());
275 } else if (!InitElement(*offsetSymbol
, designator
, scope
)) {
281 return folder
.isEmpty();
284 template <typename DSV
>
285 std::optional
<std::pair
<SomeExpr
, bool>>
286 DataInitializationCompiler
<DSV
>::ConvertElement(
287 const SomeExpr
&expr
, const evaluate::DynamicType
&type
) {
288 if (auto converted
{evaluate::ConvertToType(type
, SomeExpr
{expr
})}) {
289 return {std::make_pair(std::move(*converted
), false)};
291 // Allow DATA initialization with Hollerith and kind=1 CHARACTER like
292 // (most) other Fortran compilers do.
293 if (auto converted
{evaluate::HollerithToBOZ(
294 exprAnalyzer_
.GetFoldingContext(), expr
, type
)}) {
295 return {std::make_pair(std::move(*converted
), true)};
297 SemanticsContext
&context
{exprAnalyzer_
.context()};
298 if (context
.IsEnabled(common::LanguageFeature::LogicalIntegerAssignment
)) {
299 if (MaybeExpr converted
{evaluate::DataConstantConversionExtension(
300 exprAnalyzer_
.GetFoldingContext(), type
, expr
)}) {
301 context
.Warn(common::LanguageFeature::LogicalIntegerAssignment
,
302 exprAnalyzer_
.GetFoldingContext().messages().at(),
303 "nonstandard usage: initialization of %s with %s"_port_en_US
,
304 type
.AsFortran(), expr
.GetType().value().AsFortran());
305 return {std::make_pair(std::move(*converted
), false)};
311 template <typename DSV
>
312 bool DataInitializationCompiler
<DSV
>::InitElement(
313 const evaluate::OffsetSymbol
&offsetSymbol
, const SomeExpr
&designator
,
314 const Scope
&scope
) {
315 const Symbol
&symbol
{offsetSymbol
.symbol()};
316 const Symbol
*lastSymbol
{GetLastSymbol(designator
)};
317 bool isPointer
{lastSymbol
&& IsPointer(*lastSymbol
)};
318 bool isProcPointer
{lastSymbol
&& IsProcedurePointer(*lastSymbol
)};
319 evaluate::FoldingContext
&context
{exprAnalyzer_
.GetFoldingContext()};
321 const auto DescribeElement
{[&]() {
322 if (auto badDesignator
{
323 evaluate::OffsetToDesignator(context
, offsetSymbol
)}) {
324 return badDesignator
->AsFortran();
328 llvm::raw_string_ostream ss
{buf
};
329 ss
<< offsetSymbol
.symbol().name() << " offset " << offsetSymbol
.offset()
330 << " bytes for " << offsetSymbol
.size() << " bytes";
334 const auto GetImage
{[&]() -> evaluate::InitialImage
& {
335 // This could be (and was) written to always call std::map<>::emplace(),
336 // which should handle duplicate entries gracefully, but it was still
337 // causing memory allocation & deallocation with gcc.
338 auto iter
{inits_
.find(&symbol
)};
339 if (iter
== inits_
.end()) {
340 iter
= inits_
.emplace(&symbol
, symbol
.size()).first
;
342 auto &symbolInit
{iter
->second
};
343 symbolInit
.NoteInitializedRange(offsetSymbol
);
344 return symbolInit
.image
;
346 const auto OutOfRangeError
{[&]() {
347 evaluate::AttachDeclaration(
348 exprAnalyzer_
.context().Say(
349 "DATA statement designator '%s' is out of range for its variable '%s'"_err_en_US
,
350 DescribeElement(), symbol
.name()),
354 if (values_
.hasFatalError()) {
356 } else if (values_
.IsAtEnd()) {
357 exprAnalyzer_
.context().Say(
358 "DATA statement set has no value for '%s'"_err_en_US
,
361 } else if (static_cast<std::size_t>(
362 offsetSymbol
.offset() + offsetSymbol
.size()) > symbol
.size()) {
367 auto &messages
{context
.messages()};
369 messages
.SetLocation(values_
.LocateSource().value_or(messages
.at()))};
370 const SomeExpr
*expr
{*values_
};
372 CHECK(exprAnalyzer_
.context().AnyFatalError());
373 } else if (symbol
.size() > maxDataInitBytes
) {
374 evaluate::AttachDeclaration(
375 exprAnalyzer_
.context().Say(
376 "'%s' is too large to initialize with a DATA statement"_todo_en_US
,
380 } else if (isPointer
) {
381 if (static_cast<std::size_t>(offsetSymbol
.offset() + offsetSymbol
.size()) >
384 } else if (evaluate::IsNullPointer(*expr
)) {
385 // nothing to do; rely on zero initialization
387 } else if (isProcPointer
) {
388 if (evaluate::IsProcedureDesignator(*expr
)) {
389 if (CheckPointerAssignment(exprAnalyzer_
.context(), designator
, *expr
,
391 /*isBoundsRemapping=*/false, /*isAssumedRank=*/false)) {
392 if (lastSymbol
->has
<ProcEntityDetails
>()) {
393 GetImage().AddPointer(offsetSymbol
.offset(), *expr
);
396 evaluate::AttachDeclaration(
397 exprAnalyzer_
.context().Say(
398 "DATA statement initialization of procedure pointer '%s' declared using a POINTER statement and an INTERFACE instead of a PROCEDURE statement"_todo_en_US
,
405 "Data object '%s' may not be used to initialize '%s', which is a procedure pointer"_err_en_US
,
406 expr
->AsFortran(), DescribeElement());
408 } else if (evaluate::IsProcedure(*expr
)) {
410 "Procedure '%s' may not be used to initialize '%s', which is not a procedure pointer"_err_en_US
,
411 expr
->AsFortran(), DescribeElement());
412 } else if (CheckInitialDataPointerTarget(
413 exprAnalyzer_
.context(), designator
, *expr
, scope
)) {
414 GetImage().AddPointer(offsetSymbol
.offset(), *expr
);
417 } else if (evaluate::IsNullPointer(*expr
)) {
418 exprAnalyzer_
.Say("Initializer for '%s' must not be a pointer"_err_en_US
,
420 } else if (evaluate::IsProcedureDesignator(*expr
)) {
421 exprAnalyzer_
.Say("Initializer for '%s' must not be a procedure"_err_en_US
,
423 } else if (auto designatorType
{designator
.GetType()}) {
424 if (expr
->Rank() > 0) {
425 // Because initial-data-target is ambiguous with scalar-constant and
426 // scalar-constant-subobject at parse time, enforcement of scalar-*
427 // must be deferred to here.
429 "DATA statement value initializes '%s' with an array"_err_en_US
,
431 } else if (auto converted
{ConvertElement(*expr
, *designatorType
)}) {
432 // value non-pointer initialization
433 if (IsBOZLiteral(*expr
) &&
434 designatorType
->category() != TypeCategory::Integer
) { // 8.6.7(11)
435 exprAnalyzer_
.Warn(common::LanguageFeature::DataStmtExtensions
,
436 "BOZ literal should appear in a DATA statement only as a value for an integer object, but '%s' is '%s'"_port_en_US
,
437 DescribeElement(), designatorType
->AsFortran());
438 } else if (converted
->second
) {
439 exprAnalyzer_
.Warn(common::LanguageFeature::DataStmtExtensions
,
440 "DATA statement value initializes '%s' of type '%s' with CHARACTER"_port_en_US
,
441 DescribeElement(), designatorType
->AsFortran());
443 auto folded
{evaluate::Fold(context
, std::move(converted
->first
))};
444 // Rewritten from a switch() in order to avoid getting complaints
445 // about a missing "default:" from some compilers and complaints
446 // about a redundant "default:" from others.
447 auto status
{GetImage().Add(
448 offsetSymbol
.offset(), offsetSymbol
.size(), folded
, context
)};
449 if (status
== evaluate::InitialImage::Ok
) {
451 } else if (status
== evaluate::InitialImage::NotAConstant
) {
453 "DATA statement value '%s' for '%s' is not a constant"_err_en_US
,
454 folded
.AsFortran(), DescribeElement());
455 } else if (status
== evaluate::InitialImage::OutOfRange
) {
457 } else if (status
== evaluate::InitialImage::LengthMismatch
) {
458 exprAnalyzer_
.Warn(common::UsageWarning::DataLength
,
459 "DATA statement value '%s' for '%s' has the wrong length"_warn_en_US
,
460 folded
.AsFortran(), DescribeElement());
462 } else if (status
== evaluate::InitialImage::TooManyElems
) {
463 exprAnalyzer_
.Say("DATA statement has too many elements"_err_en_US
);
465 CHECK(exprAnalyzer_
.context().AnyFatalError());
468 exprAnalyzer_
.context().Say(
469 "DATA statement value could not be converted to the type '%s' of the object '%s'"_err_en_US
,
470 designatorType
->AsFortran(), DescribeElement());
473 CHECK(exprAnalyzer_
.context().AnyFatalError());
478 void AccumulateDataInitializations(DataInitializations
&inits
,
479 evaluate::ExpressionAnalyzer
&exprAnalyzer
,
480 const parser::DataStmtSet
&set
) {
481 DataInitializationCompiler scanner
{
482 inits
, exprAnalyzer
, std::get
<std::list
<parser::DataStmtValue
>>(set
.t
)};
483 for (const auto &object
:
484 std::get
<std::list
<parser::DataStmtObject
>>(set
.t
)) {
485 if (!scanner
.Scan(object
)) {
489 if (scanner
.HasSurplusValues()) {
490 exprAnalyzer
.context().Say(
491 "DATA statement set has more values than objects"_err_en_US
);
495 void AccumulateDataInitializations(DataInitializations
&inits
,
496 evaluate::ExpressionAnalyzer
&exprAnalyzer
, const Symbol
&symbol
,
497 const std::list
<common::Indirection
<parser::DataStmtValue
>> &list
) {
498 DataInitializationCompiler
<common::Indirection
<parser::DataStmtValue
>>
499 scanner
{inits
, exprAnalyzer
, list
};
500 if (scanner
.Scan(symbol
) && scanner
.HasSurplusValues()) {
501 exprAnalyzer
.context().Say(
502 "DATA statement set has more values than objects"_err_en_US
);
506 // Looks for default derived type component initialization -- but
507 // *not* allocatables.
508 static const DerivedTypeSpec
*HasDefaultInitialization(const Symbol
&symbol
) {
509 if (const auto *object
{symbol
.detailsIf
<ObjectEntityDetails
>()}) {
510 if (object
->init().has_value()) {
511 return nullptr; // init is explicit, not default
512 } else if (!object
->isDummy() && object
->type()) {
513 if (const DerivedTypeSpec
* derived
{object
->type()->AsDerived()}) {
514 DirectComponentIterator directs
{*derived
};
515 if (llvm::any_of(directs
, [](const Symbol
&component
) {
516 return !IsAllocatable(component
) &&
517 HasDeclarationInitializer(component
);
527 // PopulateWithComponentDefaults() adds initializations to an instance
528 // of SymbolDataInitialization containing all of the default component
531 static void PopulateWithComponentDefaults(SymbolDataInitialization
&init
,
532 std::size_t offset
, const DerivedTypeSpec
&derived
,
533 evaluate::FoldingContext
&foldingContext
);
535 static void PopulateWithComponentDefaults(SymbolDataInitialization
&init
,
536 std::size_t offset
, const DerivedTypeSpec
&derived
,
537 evaluate::FoldingContext
&foldingContext
, const Symbol
&symbol
) {
538 if (auto extents
{evaluate::GetConstantExtents(foldingContext
, symbol
)}) {
539 const Scope
&scope
{derived
.scope() ? *derived
.scope()
540 : DEREF(derived
.typeSymbol().scope())};
541 std::size_t stride
{scope
.size()};
542 if (std::size_t alignment
{scope
.alignment().value_or(0)}) {
543 stride
= ((stride
+ alignment
- 1) / alignment
) * alignment
;
545 for (auto elements
{evaluate::GetSize(*extents
)}; elements
-- > 0;
547 PopulateWithComponentDefaults(init
, offset
, derived
, foldingContext
);
552 // F'2018 19.5.3(10) allows storage-associated default component initialization
553 // when the values are identical.
554 static void PopulateWithComponentDefaults(SymbolDataInitialization
&init
,
555 std::size_t offset
, const DerivedTypeSpec
&derived
,
556 evaluate::FoldingContext
&foldingContext
) {
558 derived
.scope() ? *derived
.scope() : DEREF(derived
.typeSymbol().scope())};
559 for (const auto &pair
: scope
) {
560 const Symbol
&component
{*pair
.second
};
561 std::size_t componentOffset
{offset
+ component
.offset()};
562 if (const auto *object
{component
.detailsIf
<ObjectEntityDetails
>()}) {
563 if (!IsAllocatable(component
) && !IsAutomatic(component
)) {
564 bool initialized
{false};
565 if (object
->init()) {
567 if (IsPointer(component
)) {
568 if (auto extant
{init
.image
.AsConstantPointer(componentOffset
)}) {
569 initialized
= !(*extant
== *object
->init());
572 init
.image
.AddPointer(componentOffset
, *object
->init());
574 } else { // data, not pointer
575 if (auto dyType
{evaluate::DynamicType::From(component
)}) {
576 if (auto extents
{evaluate::GetConstantExtents(
577 foldingContext
, component
)}) {
578 if (auto extant
{init
.image
.AsConstant(foldingContext
, *dyType
,
579 std::nullopt
, *extents
, false /*don't pad*/,
581 initialized
= !(*extant
== *object
->init());
586 init
.image
.Add(componentOffset
, component
.size(), *object
->init(),
590 } else if (const DeclTypeSpec
* type
{component
.GetType()}) {
591 if (const DerivedTypeSpec
* componentDerived
{type
->AsDerived()}) {
592 PopulateWithComponentDefaults(init
, componentOffset
,
593 *componentDerived
, foldingContext
, component
);
597 init
.NoteInitializedRange(componentOffset
, component
.size());
600 } else if (const auto *proc
{component
.detailsIf
<ProcEntityDetails
>()}) {
601 if (proc
->init() && *proc
->init()) {
602 SomeExpr procPtrInit
{evaluate::ProcedureDesignator
{**proc
->init()}};
603 auto extant
{init
.image
.AsConstantPointer(componentOffset
)};
604 if (!extant
|| !(*extant
== procPtrInit
)) {
605 init
.NoteInitializedRange(componentOffset
, component
.size());
606 init
.image
.AddPointer(componentOffset
, std::move(procPtrInit
));
613 static bool CheckForOverlappingInitialization(
614 const std::list
<SymbolRef
> &symbols
,
615 SymbolDataInitialization
&initialization
,
616 evaluate::ExpressionAnalyzer
&exprAnalyzer
, const std::string
&what
) {
618 auto &context
{exprAnalyzer
.GetFoldingContext()};
619 initialization
.initializedRanges
.sort();
620 ConstantSubscript next
{0};
621 for (const auto &range
: initialization
.initializedRanges
) {
622 if (range
.start() < next
) {
623 result
= false; // error: overlap
625 for (const Symbol
&symbol
: symbols
) {
626 auto offset
{range
.start() -
627 static_cast<ConstantSubscript
>(
628 symbol
.offset() - symbols
.front()->offset())};
630 if (auto badDesignator
{evaluate::OffsetToDesignator(
631 context
, symbol
, offset
, range
.size())}) {
633 exprAnalyzer
.Say(symbol
.name(),
634 "%s affect '%s' more than once"_err_en_US
, what
,
635 badDesignator
->AsFortran());
641 next
= range
.start() + range
.size();
642 CHECK(next
<= static_cast<ConstantSubscript
>(initialization
.image
.size()));
647 static void IncorporateExplicitInitialization(
648 SymbolDataInitialization
&combined
, DataInitializations
&inits
,
649 const Symbol
&symbol
, ConstantSubscript firstOffset
,
650 evaluate::FoldingContext
&foldingContext
) {
651 auto iter
{inits
.find(&symbol
)};
652 const auto offset
{symbol
.offset() - firstOffset
};
653 if (iter
!= inits
.end()) { // DATA statement initialization
654 for (const auto &range
: iter
->second
.initializedRanges
) {
655 auto at
{offset
+ range
.start()};
656 combined
.NoteInitializedRange(at
, range
.size());
657 combined
.image
.Incorporate(
658 at
, iter
->second
.image
, range
.start(), range
.size());
660 if (removeOriginalInits
) {
663 } else { // Declaration initialization
664 Symbol
&mutableSymbol
{const_cast<Symbol
&>(symbol
)};
665 if (IsPointer(mutableSymbol
)) {
666 if (auto *object
{mutableSymbol
.detailsIf
<ObjectEntityDetails
>()}) {
667 if (object
->init()) {
668 combined
.NoteInitializedRange(offset
, mutableSymbol
.size());
669 combined
.image
.AddPointer(offset
, *object
->init());
670 if (removeOriginalInits
) {
671 object
->init().reset();
674 } else if (auto *proc
{mutableSymbol
.detailsIf
<ProcEntityDetails
>()}) {
675 if (proc
->init() && *proc
->init()) {
676 combined
.NoteInitializedRange(offset
, mutableSymbol
.size());
677 combined
.image
.AddPointer(
678 offset
, SomeExpr
{evaluate::ProcedureDesignator
{**proc
->init()}});
679 if (removeOriginalInits
) {
680 proc
->init().reset();
684 } else if (auto *object
{mutableSymbol
.detailsIf
<ObjectEntityDetails
>()}) {
685 if (!IsNamedConstant(mutableSymbol
) && object
->init()) {
686 combined
.NoteInitializedRange(offset
, mutableSymbol
.size());
688 offset
, mutableSymbol
.size(), *object
->init(), foldingContext
);
689 if (removeOriginalInits
) {
690 object
->init().reset();
697 // Finds the size of the smallest element type in a list of
698 // storage-associated objects.
699 static std::size_t ComputeMinElementBytes(
700 const std::list
<SymbolRef
> &associated
,
701 evaluate::FoldingContext
&foldingContext
) {
702 std::size_t minElementBytes
{1};
703 const Symbol
&first
{*associated
.front()};
704 for (const Symbol
&s
: associated
) {
705 if (auto dyType
{evaluate::DynamicType::From(s
)}) {
706 auto size
{static_cast<std::size_t>(
707 evaluate::ToInt64(dyType
->MeasureSizeInBytes(foldingContext
, true))
709 if (std::size_t alignment
{
710 dyType
->GetAlignment(foldingContext
.targetCharacteristics())}) {
711 size
= ((size
+ alignment
- 1) / alignment
) * alignment
;
714 minElementBytes
= size
;
716 minElementBytes
= std::min(minElementBytes
, size
);
722 return minElementBytes
;
725 // Checks for overlapping initialization errors in a list of
726 // storage-associated objects. Default component initializations
727 // are allowed to be overridden by explicit initializations.
728 // If the objects are static, save the combined initializer as
729 // a compiler-created object that covers all of them.
730 static bool CombineEquivalencedInitialization(
731 const std::list
<SymbolRef
> &associated
,
732 evaluate::ExpressionAnalyzer
&exprAnalyzer
, DataInitializations
&inits
) {
733 // Compute the minimum common granularity and total size
734 const Symbol
&first
{*associated
.front()};
735 std::size_t maxLimit
{0};
736 for (const Symbol
&s
: associated
) {
737 CHECK(s
.offset() >= first
.offset());
738 auto limit
{s
.offset() + s
.size()};
739 if (limit
> maxLimit
) {
743 auto bytes
{static_cast<common::ConstantSubscript
>(maxLimit
- first
.offset())};
744 Scope
&scope
{const_cast<Scope
&>(first
.owner())};
745 // Combine the initializations of the associated objects.
746 // Apply all default initializations first.
747 SymbolDataInitialization combined
{static_cast<std::size_t>(bytes
)};
748 auto &foldingContext
{exprAnalyzer
.GetFoldingContext()};
749 for (const Symbol
&s
: associated
) {
750 if (!IsNamedConstant(s
)) {
751 if (const auto *derived
{HasDefaultInitialization(s
)}) {
752 PopulateWithComponentDefaults(
753 combined
, s
.offset() - first
.offset(), *derived
, foldingContext
, s
);
757 if (!CheckForOverlappingInitialization(associated
, combined
, exprAnalyzer
,
758 "Distinct default component initializations of equivalenced objects"s
)) {
761 // Don't complain about overlap between explicit initializations and
762 // default initializations.
763 combined
.initializedRanges
.clear();
764 // Now overlay all explicit initializations from DATA statements and
765 // from initializers in declarations.
766 for (const Symbol
&symbol
: associated
) {
767 IncorporateExplicitInitialization(
768 combined
, inits
, symbol
, first
.offset(), foldingContext
);
770 if (!CheckForOverlappingInitialization(associated
, combined
, exprAnalyzer
,
771 "Explicit initializations of equivalenced objects"s
)) {
774 // If the items are in static storage, save the final initialization.
775 if (llvm::any_of(associated
, [](SymbolRef ref
) { return IsSaved(*ref
); })) {
776 // Create a compiler array temp that overlaps all the items.
777 SourceName name
{exprAnalyzer
.context().GetTempName(scope
)};
779 scope
.try_emplace(name
, Attrs
{Attr::SAVE
}, ObjectEntityDetails
{})};
780 CHECK(emplaced
.second
);
781 Symbol
&combinedSymbol
{*emplaced
.first
->second
};
782 combinedSymbol
.set(Symbol::Flag::CompilerCreated
);
783 inits
.emplace(&combinedSymbol
, std::move(combined
));
784 auto &details
{combinedSymbol
.get
<ObjectEntityDetails
>()};
785 combinedSymbol
.set_offset(first
.offset());
786 combinedSymbol
.set_size(bytes
);
787 std::size_t minElementBytes
{
788 ComputeMinElementBytes(associated
, foldingContext
)};
789 if (!exprAnalyzer
.GetFoldingContext().targetCharacteristics().IsTypeEnabled(
790 TypeCategory::Integer
, minElementBytes
) ||
791 (bytes
% minElementBytes
) != 0) {
794 const DeclTypeSpec
&typeSpec
{scope
.MakeNumericType(
795 TypeCategory::Integer
, KindExpr
{minElementBytes
})};
796 details
.set_type(typeSpec
);
798 arraySpec
.emplace_back(ShapeSpec::MakeExplicit(Bound
{
799 bytes
/ static_cast<common::ConstantSubscript
>(minElementBytes
)}));
800 details
.set_shape(arraySpec
);
801 if (const auto *commonBlock
{FindCommonBlockContaining(first
)}) {
802 details
.set_commonBlock(*commonBlock
);
804 // Add an EQUIVALENCE set to the scope so that the new object appears in
805 // the results of GetStorageAssociations().
806 auto &newSet
{scope
.equivalenceSets().emplace_back()};
807 newSet
.emplace_back(combinedSymbol
);
808 newSet
.emplace_back(const_cast<Symbol
&>(first
));
813 // When a statically-allocated derived type variable has no explicit
814 // initialization, but its type has at least one nonallocatable ultimate
815 // component with default initialization, make its initialization explicit.
816 [[maybe_unused
]] static void MakeDefaultInitializationExplicit(
817 const Scope
&scope
, const std::list
<std::list
<SymbolRef
>> &associations
,
818 evaluate::FoldingContext
&foldingContext
, DataInitializations
&inits
) {
819 UnorderedSymbolSet equivalenced
;
820 for (const std::list
<SymbolRef
> &association
: associations
) {
821 for (const Symbol
&symbol
: association
) {
822 equivalenced
.emplace(symbol
);
825 for (const auto &pair
: scope
) {
826 const Symbol
&symbol
{*pair
.second
};
827 if (!symbol
.test(Symbol::Flag::InDataStmt
) &&
828 !HasDeclarationInitializer(symbol
) && IsSaved(symbol
) &&
829 equivalenced
.find(symbol
) == equivalenced
.end()) {
830 // Static object, no local storage association, no explicit initialization
831 if (const DerivedTypeSpec
* derived
{HasDefaultInitialization(symbol
)}) {
832 auto newInitIter
{inits
.emplace(&symbol
, symbol
.size())};
833 CHECK(newInitIter
.second
);
834 auto &newInit
{newInitIter
.first
->second
};
835 PopulateWithComponentDefaults(
836 newInit
, 0, *derived
, foldingContext
, symbol
);
842 // Traverses the Scopes to:
843 // 1) combine initialization of equivalenced objects, &
844 // 2) optionally make initialization explicit for otherwise uninitialized static
845 // objects of derived types with default component initialization
846 // Returns false on error.
847 static bool ProcessScopes(const Scope
&scope
,
848 evaluate::ExpressionAnalyzer
&exprAnalyzer
, DataInitializations
&inits
) {
849 bool result
{true}; // no error
850 switch (scope
.kind()) {
851 case Scope::Kind::Global
:
852 case Scope::Kind::Module
:
853 case Scope::Kind::MainProgram
:
854 case Scope::Kind::Subprogram
:
855 case Scope::Kind::BlockData
:
856 case Scope::Kind::BlockConstruct
: {
857 std::list
<std::list
<SymbolRef
>> associations
{GetStorageAssociations(scope
)};
858 for (const std::list
<SymbolRef
> &associated
: associations
) {
859 if (std::find_if(associated
.begin(), associated
.end(), [](SymbolRef ref
) {
860 return IsInitialized(*ref
);
861 }) != associated
.end()) {
863 CombineEquivalencedInitialization(associated
, exprAnalyzer
, inits
);
866 if constexpr (makeDefaultInitializationExplicit
) {
867 MakeDefaultInitializationExplicit(
868 scope
, associations
, exprAnalyzer
.GetFoldingContext(), inits
);
870 for (const Scope
&child
: scope
.children()) {
871 result
&= ProcessScopes(child
, exprAnalyzer
, inits
);
879 // Converts the static initialization image for a single symbol with
880 // one or more DATA statement appearances.
881 void ConstructInitializer(const Symbol
&symbol
,
882 SymbolDataInitialization
&initialization
,
883 evaluate::ExpressionAnalyzer
&exprAnalyzer
) {
884 std::list
<SymbolRef
> symbols
{symbol
};
885 CheckForOverlappingInitialization(
886 symbols
, initialization
, exprAnalyzer
, "DATA statement initializations"s
);
887 auto &context
{exprAnalyzer
.GetFoldingContext()};
888 if (const auto *proc
{symbol
.detailsIf
<ProcEntityDetails
>()}) {
889 CHECK(IsProcedurePointer(symbol
));
890 auto &mutableProc
{const_cast<ProcEntityDetails
&>(*proc
)};
891 if (MaybeExpr expr
{initialization
.image
.AsConstantPointer()}) {
892 if (const auto *procDesignator
{
893 std::get_if
<evaluate::ProcedureDesignator
>(&expr
->u
)}) {
894 CHECK(!procDesignator
->GetComponent());
895 if (const auto *intrin
{procDesignator
->GetSpecificIntrinsic()}) {
896 const Symbol
*intrinSymbol
{
897 symbol
.owner().FindSymbol(SourceName
{intrin
->name
})};
898 mutableProc
.set_init(DEREF(intrinSymbol
));
900 mutableProc
.set_init(DEREF(procDesignator
->GetSymbol()));
903 CHECK(evaluate::IsNullProcedurePointer(*expr
));
904 mutableProc
.set_init(nullptr);
907 mutableProc
.set_init(nullptr);
909 } else if (const auto *object
{symbol
.detailsIf
<ObjectEntityDetails
>()}) {
910 auto &mutableObject
{const_cast<ObjectEntityDetails
&>(*object
)};
911 if (IsPointer(symbol
)) {
912 if (auto ptr
{initialization
.image
.AsConstantPointer()}) {
913 mutableObject
.set_init(*ptr
);
915 mutableObject
.set_init(SomeExpr
{evaluate::NullPointer
{}});
917 } else if (auto symbolType
{evaluate::DynamicType::From(symbol
)}) {
918 if (auto extents
{evaluate::GetConstantExtents(context
, symbol
)}) {
919 mutableObject
.set_init(initialization
.image
.AsConstant(
920 context
, *symbolType
, std::nullopt
, *extents
));
922 exprAnalyzer
.Say(symbol
.name(),
923 "internal: unknown shape for '%s' while constructing initializer from DATA"_err_en_US
,
928 exprAnalyzer
.Say(symbol
.name(),
929 "internal: no type for '%s' while constructing initializer from DATA"_err_en_US
,
933 if (!object
->init()) {
934 exprAnalyzer
.Say(symbol
.name(),
935 "internal: could not construct an initializer from DATA statements for '%s'"_err_en_US
,
939 CHECK(exprAnalyzer
.context().AnyFatalError());
943 void ConvertToInitializers(
944 DataInitializations
&inits
, evaluate::ExpressionAnalyzer
&exprAnalyzer
) {
946 exprAnalyzer
.context().globalScope(), exprAnalyzer
, inits
)) {
947 for (auto &[symbolPtr
, initialization
] : inits
) {
948 ConstructInitializer(*symbolPtr
, initialization
, exprAnalyzer
);
952 } // namespace Fortran::semantics