/* * Copyright (c) 2011, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package com.oracle.max.graal.nodes.extended; import com.oracle.max.graal.graph.*; import com.oracle.max.graal.nodes.*; import com.oracle.max.graal.nodes.spi.*; import com.sun.cri.ci.*; import com.sun.cri.ci.CiAddress.Scale; public final class IndexedLocationNode extends LocationNode implements LIRLowerable, Canonicalizable { @Input private ValueNode index; @Data private boolean indexScalingEnabled; public ValueNode index() { return index; } public static Object getArrayLocation(CiKind elementKind) { return elementKind; } /** * @return whether scaling of the index by the value kind's size is enabled (the default) or disabled. */ public boolean indexScalingEnabled() { return indexScalingEnabled; } /** * Enables or disables scaling of the index by the value kind's size. Has no effect if the index input is not used. */ public void setIndexScalingEnabled(boolean enable) { this.indexScalingEnabled = enable; } public static IndexedLocationNode create(Object identity, CiKind kind, int displacement, ValueNode index, Graph graph) { return create(identity, kind, displacement, index, graph, true); } public static IndexedLocationNode create(Object identity, CiKind kind, int displacement, ValueNode index, Graph graph, boolean indexScalingEnabled) { return graph.unique(new IndexedLocationNode(identity, kind, index, displacement, indexScalingEnabled)); } private IndexedLocationNode(Object identity, CiKind kind, ValueNode index, int displacement, boolean indexScalingEnabled) { super(identity, kind, displacement); this.index = index; this.indexScalingEnabled = indexScalingEnabled; } @Override public CiAddress createAddress(LIRGeneratorTool gen, ValueNode object) { CiValue base = gen.operand(object); if (base.isConstant() && ((CiConstant) base).isNull()) { base = CiValue.IllegalValue; } CiValue indexValue = gen.operand(index()); Scale indexScale = Scale.Times1; if (indexScalingEnabled) { indexScale = Scale.fromInt(gen.target().sizeInBytes(getValueKind())); } return new CiAddress(getValueKind(), base, indexValue, indexScale, displacement()); } @Override public Node canonical(CanonicalizerTool tool) { CiConstant constantIndex = index.asConstant(); if (constantIndex != null && constantIndex.kind.stackKind().isInt()) { long constantIndexLong = constantIndex.asInt(); if (indexScalingEnabled && tool.target() != null) { constantIndexLong *= tool.target().sizeInBytes(getValueKind()); } constantIndexLong += displacement(); int constantIndexInt = (int) constantIndexLong; if (constantIndexLong == constantIndexInt) { return LocationNode.create(locationIdentity(), getValueKind(), constantIndexInt, graph()); } } return this; } }