1 /*
2 * Copyright (c) 2021-2025 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "etsArrayType.h"
17 #include <cstddef>
18
19 #include "varbinder/variable.h"
20 #include "checker/ETSchecker.h"
21 #include "checker/ets/conversion.h"
22 #include "checker/types/typeRelation.h"
23 #include "checker/types/ets/etsTupleType.h"
24
25 namespace ark::es2panda::checker {
ToString(std::stringstream & ss,bool precise) const26 void ETSArrayType::ToString(std::stringstream &ss, bool precise) const
27 {
28 if (HasTypeFlag(TypeFlag::READONLY)) {
29 ss << "readonly ";
30 }
31 ss << "FixedArray<";
32 bool needParens = (element_->IsETSUnionType() || element_->IsETSFunctionType());
33 if (needParens) {
34 ss << "(";
35 }
36 element_->ToString(ss, precise);
37 if (needParens) {
38 ss << ")";
39 }
40 ss << ">";
41 }
42
ToAssemblerType(std::stringstream & ss) const43 void ETSArrayType::ToAssemblerType(std::stringstream &ss) const
44 {
45 element_->ToAssemblerType(ss);
46 }
47
ToAssemblerTypeWithRank(std::stringstream & ss) const48 void ETSArrayType::ToAssemblerTypeWithRank(std::stringstream &ss) const
49 {
50 element_->ToAssemblerType(ss);
51
52 for (uint32_t i = Rank(); i > 0; --i) {
53 ss << "[]";
54 }
55 }
56
ToDebugInfoType(std::stringstream & ss) const57 void ETSArrayType::ToDebugInfoType(std::stringstream &ss) const
58 {
59 ss << "[";
60 element_->ToDebugInfoType(ss);
61 }
62
Rank() const63 uint32_t ETSArrayType::Rank() const
64 {
65 uint32_t rank = 1;
66 const auto *iter = element_;
67 while (iter->IsETSArrayType() && iter->AsETSArrayType()->ElementType() != iter) {
68 iter = iter->AsETSArrayType()->ElementType();
69 rank++;
70 }
71
72 return rank;
73 }
74
Identical(TypeRelation * relation,Type * other)75 void ETSArrayType::Identical(TypeRelation *relation, Type *other)
76 {
77 if (other->IsETSArrayType()) {
78 // will be removed, if wildcard type is assigned to array type, not element type
79 if (element_->IsWildcardType() || other->AsETSArrayType()->ElementType()->IsWildcardType()) {
80 relation->Result(true);
81 return;
82 }
83 relation->IsIdenticalTo(element_, other->AsETSArrayType()->ElementType());
84 }
85 }
86
IsFixedArrayDeclaration(ir::AstNode * node)87 static bool IsFixedArrayDeclaration(ir::AstNode *node)
88 {
89 return node != nullptr && (node->IsArrayExpression() || node->IsETSNewArrayInstanceExpression() ||
90 node->IsETSNewMultiDimArrayInstanceExpression());
91 }
92
AssignmentSource(TypeRelation * relation,Type * target)93 bool ETSArrayType::AssignmentSource(TypeRelation *relation, Type *target)
94 {
95 if (target->IsETSResizableArrayType() && IsFixedArrayDeclaration(relation->GetNode())) {
96 relation->IsAssignableTo(element_, target->AsETSResizableArrayType()->ElementType());
97 // For lowering purpose
98 relation->GetNode()->SetTsType(target);
99 }
100 return relation->IsTrue();
101 }
102
AssignmentTarget(TypeRelation * relation,Type * source)103 void ETSArrayType::AssignmentTarget(TypeRelation *relation, Type *source)
104 {
105 if (source->HasTypeFlag(TypeFlag::READONLY)) {
106 relation->Result(false);
107 return;
108 }
109 if (source->IsETSArrayType()) {
110 if (ElementType()->IsETSPrimitiveOrEnumType() ||
111 source->AsETSArrayType()->ElementType()->IsETSPrimitiveOrEnumType()) {
112 return;
113 }
114 relation->IsAssignableTo(source->AsETSArrayType()->ElementType(), element_);
115 }
116 }
117
Cast(TypeRelation * const relation,Type * const target)118 void ETSArrayType::Cast(TypeRelation *const relation, Type *const target)
119 {
120 if (relation->IsSupertypeOf(this, target)) {
121 relation->RemoveFlags(TypeRelationFlag::UNCHECKED_CAST);
122 return;
123 }
124
125 if (target->HasTypeFlag(TypeFlag::ETS_ARRAY)) {
126 conversion::NarrowingReference(relation, this, target->AsETSArrayType());
127 if (relation->IsTrue()) {
128 return;
129 }
130
131 if (ElementType()->IsETSTypeParameter()) {
132 // unchecked cast!
133 relation->Result(true);
134 return;
135 }
136 relation->Result(relation->InCastingContext());
137 return;
138 }
139
140 relation->Result(relation->InCastingContext());
141 }
142
IsSupertypeOf(TypeRelation * const relation,Type * source)143 void ETSArrayType::IsSupertypeOf(TypeRelation *const relation, Type *source)
144 {
145 if (source->IsETSArrayType()) {
146 relation->IsSupertypeOf(this->AsETSArrayType()->ElementType(), source->AsETSArrayType()->ElementType());
147 }
148 }
149
IsSubtypeOf(TypeRelation * const relation,Type * target)150 void ETSArrayType::IsSubtypeOf(TypeRelation *const relation, Type *target)
151 {
152 if (target->IsETSObjectType() && target->AsETSObjectType()->IsGlobalETSObjectType()) {
153 relation->Result(true);
154 return;
155 }
156 }
157
Instantiate(ArenaAllocator * allocator,TypeRelation * relation,GlobalTypesHolder * globalTypes)158 Type *ETSArrayType::Instantiate(ArenaAllocator *allocator, TypeRelation *relation, GlobalTypesHolder *globalTypes)
159 {
160 auto *elementType = element_->Instantiate(allocator, relation, globalTypes);
161
162 // Some TypeFlag such as READONLY may pollute the ETSArrayType in the cache
163 ETSArrayType *arrayType =
164 relation->GetChecker()->AsETSChecker()->CreateETSArrayType(elementType, HasTypeFlag(TypeFlag::READONLY));
165 arrayType->typeFlags_ = typeFlags_;
166 return arrayType;
167 }
168
Substitute(TypeRelation * relation,const Substitution * substitution)169 Type *ETSArrayType::Substitute(TypeRelation *relation, const Substitution *substitution)
170 {
171 if (substitution == nullptr || substitution->empty()) {
172 return this;
173 }
174
175 auto *resultElt = element_->Substitute(relation, substitution);
176
177 if (resultElt == element_) {
178 return this;
179 }
180
181 // Some TypeFlag such as READONLY may pollute the ETSArrayType in the cache
182 ETSArrayType *result =
183 relation->GetChecker()->AsETSChecker()->CreateETSArrayType(resultElt, HasTypeFlag(TypeFlag::READONLY));
184 result->typeFlags_ = typeFlags_;
185 return result;
186 }
187
CheckVarianceRecursively(TypeRelation * relation,VarianceFlag varianceFlag)188 void ETSArrayType::CheckVarianceRecursively(TypeRelation *relation, VarianceFlag varianceFlag)
189 {
190 // The type of array should be Invariant
191 relation->CheckVarianceRecursively(element_, relation->TransferVariant(varianceFlag, VarianceFlag::INVARIANT));
192 }
193
194 } // namespace ark::es2panda::checker
195