blob: f57aa518e1d3aaf840f1285b83dee4aeb9e485cc [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2008 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.wst.jsdt.internal.compiler.ast;
import org.eclipse.wst.jsdt.core.ast.IASTNode;
import org.eclipse.wst.jsdt.core.ast.ILabeledStatement;
import org.eclipse.wst.jsdt.internal.compiler.ASTVisitor;
import org.eclipse.wst.jsdt.internal.compiler.flow.FlowContext;
import org.eclipse.wst.jsdt.internal.compiler.flow.FlowInfo;
import org.eclipse.wst.jsdt.internal.compiler.flow.LabelFlowContext;
import org.eclipse.wst.jsdt.internal.compiler.lookup.BlockScope;
public class LabeledStatement extends Statement implements ILabeledStatement {
public Statement statement;
public char[] label;
public int labelEnd;
// for local variables table attributes
int mergedInitStateIndex = -1;
/**
* LabeledStatement constructor comment.
*/
public LabeledStatement(char[] label, Statement statement, long labelPosition, int sourceEnd) {
this.statement = statement;
// remember useful empty statement
if (statement instanceof EmptyStatement) statement.bits |= IsUsefulEmptyStatement;
this.label = label;
this.sourceStart = (int)(labelPosition >>> 32);
this.labelEnd = (int) labelPosition;
this.sourceEnd = sourceEnd;
}
public FlowInfo analyseCode(
BlockScope currentScope,
FlowContext flowContext,
FlowInfo flowInfo) {
// need to stack a context to store explicit label, answer inits in case of normal completion merged
// with those relative to the exit path from break statement occurring inside the labeled statement.
if (statement == null) {
return flowInfo;
} else {
LabelFlowContext labelContext;
FlowInfo statementInfo, mergedInfo;
if (((statementInfo = statement
.analyseCode(
currentScope,
(labelContext =
new LabelFlowContext(
flowContext,
this,
label,
currentScope)),
flowInfo)).tagBits & FlowInfo.UNREACHABLE) != 0) {
if ((labelContext.initsOnBreak.tagBits & FlowInfo.UNREACHABLE) == 0) {
// an embedded loop has had no chance to reinject forgotten null info
mergedInfo = flowInfo.unconditionalCopy().
addInitializationsFrom(labelContext.initsOnBreak);
} else {
mergedInfo = labelContext.initsOnBreak;
}
} else {
mergedInfo = statementInfo.mergedWith(labelContext.initsOnBreak);
}
// mergedInitStateIndex =
// currentScope.methodScope().recordInitializationStates(mergedInfo);
if ((this.bits & ASTNode.LabelUsed) == 0) {
currentScope.problemReporter().unusedLabel(this);
}
return mergedInfo;
}
}
public ASTNode concreteStatement() {
// return statement.concreteStatement(); // for supporting nested labels: a:b:c: someStatement (see 21912)
return statement;
}
public StringBuffer printStatement(int tab, StringBuffer output) {
printIndent(tab, output).append(label).append(": "); //$NON-NLS-1$
if (this.statement == null)
output.append(';');
else
this.statement.printStatement(0, output);
return output;
}
public void resolve(BlockScope scope) {
if (this.statement != null) {
this.statement.resolve(scope);
}
}
public void traverse(
ASTVisitor visitor,
BlockScope blockScope) {
if (visitor.visit(this, blockScope)) {
if (this.statement != null) this.statement.traverse(visitor, blockScope);
}
visitor.endVisit(this, blockScope);
}
public int getASTType() {
return IASTNode.LABELED_STATEMENT;
}
}