|
40 | 40 | */
|
41 | 41 | package com.oracle.graal.python.builtins.objects.iterator;
|
42 | 42 |
|
| 43 | +import static com.oracle.graal.python.builtins.PythonBuiltinClassType.PIterator; |
43 | 44 | import static com.oracle.graal.python.builtins.PythonBuiltinClassType.TypeError;
|
44 | 45 | import static com.oracle.graal.python.nodes.SpecialMethodNames.__LENGTH_HINT__;
|
45 | 46 | import static com.oracle.graal.python.nodes.SpecialMethodNames.__LEN__;
|
46 | 47 |
|
| 48 | +import java.util.ArrayList; |
| 49 | +import java.util.List; |
| 50 | + |
47 | 51 | import com.oracle.graal.python.builtins.objects.PNone;
|
48 | 52 | import com.oracle.graal.python.builtins.objects.PNotImplemented;
|
49 | 53 | import com.oracle.graal.python.builtins.objects.common.SequenceNodes.GetSequenceStorageNode;
|
50 | 54 | import com.oracle.graal.python.builtins.objects.common.SequenceStorageNodes;
|
| 55 | +import com.oracle.graal.python.builtins.objects.iterator.IteratorNodesFactory.GetInternalIteratorSequenceStorageNodeGen; |
| 56 | +import com.oracle.graal.python.builtins.objects.list.PList; |
51 | 57 | import com.oracle.graal.python.builtins.objects.str.PString;
|
| 58 | +import com.oracle.graal.python.builtins.objects.tuple.PTuple; |
52 | 59 | import com.oracle.graal.python.builtins.objects.type.SpecialMethodSlot;
|
53 | 60 | import com.oracle.graal.python.lib.PyIndexCheckNode;
|
54 | 61 | import com.oracle.graal.python.lib.PyNumberAsSizeNode;
|
|
73 | 80 | import com.oracle.graal.python.runtime.sequence.storage.SequenceStorage;
|
74 | 81 | import com.oracle.truffle.api.CompilerDirectives;
|
75 | 82 | import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
|
| 83 | +import com.oracle.truffle.api.dsl.Bind; |
76 | 84 | import com.oracle.truffle.api.dsl.Cached;
|
| 85 | +import com.oracle.truffle.api.dsl.Fallback; |
77 | 86 | import com.oracle.truffle.api.dsl.GenerateNodeFactory;
|
78 | 87 | import com.oracle.truffle.api.dsl.GenerateUncached;
|
79 | 88 | import com.oracle.truffle.api.dsl.ImportStatic;
|
|
85 | 94 | import com.oracle.truffle.api.nodes.Node;
|
86 | 95 | import com.oracle.truffle.api.profiles.ConditionProfile;
|
87 | 96 | import com.oracle.truffle.api.profiles.LoopConditionProfile;
|
88 |
| -import java.util.ArrayList; |
89 |
| -import java.util.List; |
90 | 97 |
|
91 | 98 | public abstract class IteratorNodes {
|
92 | 99 |
|
@@ -181,6 +188,114 @@ int length(VirtualFrame frame, Object iterable,
|
181 | 188 | }
|
182 | 189 | }
|
183 | 190 |
|
| 191 | + @ImportStatic(PGuards.class) |
| 192 | + @GenerateUncached |
| 193 | + public abstract static class GetInternalIteratorSequenceStorage extends Node { |
| 194 | + public static GetInternalIteratorSequenceStorage getUncached() { |
| 195 | + return GetInternalIteratorSequenceStorageNodeGen.getUncached(); |
| 196 | + } |
| 197 | + |
| 198 | + /** |
| 199 | + * The argument must be a builtin iterator, which points to the first element of the |
| 200 | + * internal sequence storage. Returns {@code null} if the sequence storage is not available |
| 201 | + * or if the iterator is not pointing to the first item in the storage. |
| 202 | + */ |
| 203 | + public final SequenceStorage execute(PBuiltinIterator iterator) { |
| 204 | + assert GetClassNode.getUncached().execute(iterator) == PIterator; |
| 205 | + assert iterator.index == 0 && !iterator.isExhausted(); |
| 206 | + return executeInternal(iterator); |
| 207 | + } |
| 208 | + |
| 209 | + protected abstract SequenceStorage executeInternal(PBuiltinIterator iterator); |
| 210 | + |
| 211 | + @Specialization(guards = "isList(it.sequence)") |
| 212 | + static SequenceStorage doSequenceList(PSequenceIterator it) { |
| 213 | + return ((PList) it.sequence).getSequenceStorage(); |
| 214 | + } |
| 215 | + |
| 216 | + @Specialization |
| 217 | + static SequenceStorage doSequenceLong(PLongSequenceIterator it) { |
| 218 | + return it.sequence; |
| 219 | + } |
| 220 | + |
| 221 | + @Specialization |
| 222 | + static SequenceStorage doSequenceDouble(PDoubleSequenceIterator it) { |
| 223 | + return it.sequence; |
| 224 | + } |
| 225 | + |
| 226 | + @Specialization |
| 227 | + static SequenceStorage doSequenceObj(PObjectSequenceIterator it) { |
| 228 | + return it.sequence; |
| 229 | + } |
| 230 | + |
| 231 | + @Specialization |
| 232 | + static SequenceStorage doSequenceIntSeq(PIntegerSequenceIterator it) { |
| 233 | + return it.sequence; |
| 234 | + } |
| 235 | + |
| 236 | + @Specialization(guards = "isPTuple(it.sequence)") |
| 237 | + static SequenceStorage doSequenceTuple(PSequenceIterator it) { |
| 238 | + return ((PTuple) it.sequence).getSequenceStorage(); |
| 239 | + } |
| 240 | + |
| 241 | + @Fallback |
| 242 | + static SequenceStorage doOthers(PBuiltinIterator it) { |
| 243 | + return null; |
| 244 | + } |
| 245 | + } |
| 246 | + |
| 247 | + @ImportStatic(PGuards.class) |
| 248 | + public abstract static class BuiltinIteratorLengthHint extends Node { |
| 249 | + /** |
| 250 | + * The argument must be a builtin iterator. Returns {@code -1} if the length hint is not |
| 251 | + * available and rewrites itself to generic fallback that always returns {@code -1}. |
| 252 | + */ |
| 253 | + public final int execute(PBuiltinIterator iterator) { |
| 254 | + assert GetClassNode.getUncached().execute(iterator) == PIterator; |
| 255 | + return executeInternal(iterator); |
| 256 | + } |
| 257 | + |
| 258 | + protected abstract int executeInternal(PBuiltinIterator iterator); |
| 259 | + |
| 260 | + protected static SequenceStorage getStorage(GetInternalIteratorSequenceStorage getSeqStorage, PBuiltinIterator it) { |
| 261 | + return it.index != 0 || it.isExhausted() ? null : getSeqStorage.execute(it); |
| 262 | + } |
| 263 | + |
| 264 | + @Specialization(guards = "storage != null") |
| 265 | + static int doSeqStorage(@SuppressWarnings("unused") PBuiltinIterator it, |
| 266 | + @SuppressWarnings("unused") @Cached GetInternalIteratorSequenceStorage getSeqStorage, |
| 267 | + @Bind("getStorage(getSeqStorage, it)") SequenceStorage storage) { |
| 268 | + return ensurePositive(storage.length()); |
| 269 | + } |
| 270 | + |
| 271 | + @Specialization |
| 272 | + static int doString(PStringIterator it) { |
| 273 | + return ensurePositive(it.value.length()); |
| 274 | + } |
| 275 | + |
| 276 | + @Specialization |
| 277 | + static int doSequenceArr(PArrayIterator it) { |
| 278 | + return ensurePositive(it.array.getLength()); |
| 279 | + } |
| 280 | + |
| 281 | + @Specialization |
| 282 | + static int doSequenceIntRange(PIntRangeIterator it) { |
| 283 | + return ensurePositive(it.getLength()); |
| 284 | + } |
| 285 | + |
| 286 | + @Specialization(replaces = {"doSeqStorage", "doString", "doSequenceArr", "doSequenceIntRange"}) |
| 287 | + static int doGeneric(@SuppressWarnings("unused") PBuiltinIterator it) { |
| 288 | + return -1; |
| 289 | + } |
| 290 | + |
| 291 | + static int ensurePositive(int len) { |
| 292 | + if (len < 0) { |
| 293 | + throw CompilerDirectives.shouldNotReachHere(); |
| 294 | + } |
| 295 | + return len; |
| 296 | + } |
| 297 | + } |
| 298 | + |
184 | 299 | @GenerateUncached
|
185 | 300 | public abstract static class IsIteratorObjectNode extends Node {
|
186 | 301 |
|
|
0 commit comments