|
|
|
|
|
|
|
*/ |
|
/* |
|
* Licensed to the Apache Software Foundation (ASF) under one or more |
|
* contributor license agreements. See the NOTICE file distributed with |
|
* this work for additional information regarding copyright ownership. |
|
* The ASF licenses this file to You under the Apache License, Version 2.0 |
|
* (the "License"); you may not use this file except in compliance with |
|
* the License. You may obtain a copy of the License at |
|
* |
|
* http://www.apache.org/licenses/LICENSE-2.0 |
|
* |
|
* Unless required by applicable law or agreed to in writing, software |
|
* distributed under the License is distributed on an "AS IS" BASIS, |
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
* See the License for the specific language governing permissions and |
|
* limitations under the License. |
|
*/ |
|
|
|
package com.sun.org.apache.xalan.internal.xsltc.compiler; |
|
|
|
import com.sun.org.apache.bcel.internal.generic.ALOAD; |
|
import com.sun.org.apache.bcel.internal.generic.ASTORE; |
|
import com.sun.org.apache.bcel.internal.generic.BranchHandle; |
|
import com.sun.org.apache.bcel.internal.generic.ConstantPoolGen; |
|
import com.sun.org.apache.bcel.internal.generic.IFEQ; |
|
import com.sun.org.apache.bcel.internal.generic.IFNULL; |
|
import com.sun.org.apache.bcel.internal.generic.ILOAD; |
|
import com.sun.org.apache.bcel.internal.generic.INVOKEINTERFACE; |
|
import com.sun.org.apache.bcel.internal.generic.INVOKEVIRTUAL; |
|
import com.sun.org.apache.bcel.internal.generic.ISTORE; |
|
import com.sun.org.apache.bcel.internal.generic.InstructionHandle; |
|
import com.sun.org.apache.bcel.internal.generic.InstructionList; |
|
import com.sun.org.apache.bcel.internal.generic.LocalVariableGen; |
|
import com.sun.org.apache.xalan.internal.xsltc.compiler.util.ClassGenerator; |
|
import com.sun.org.apache.xalan.internal.xsltc.compiler.util.ErrorMsg; |
|
import com.sun.org.apache.xalan.internal.xsltc.compiler.util.MethodGenerator; |
|
import com.sun.org.apache.xalan.internal.xsltc.compiler.util.Type; |
|
import com.sun.org.apache.xalan.internal.xsltc.compiler.util.TypeCheckError; |
|
import com.sun.org.apache.xalan.internal.xsltc.compiler.util.Util; |
|
|
|
|
|
|
|
|
|
*/ |
|
final class Copy extends Instruction { |
|
private UseAttributeSets _useSets; |
|
|
|
public void parseContents(Parser parser) { |
|
final String useSets = getAttribute("use-attribute-sets"); |
|
if (useSets.length() > 0) { |
|
if (!Util.isValidQNames(useSets)) { |
|
ErrorMsg err = new ErrorMsg(ErrorMsg.INVALID_QNAME_ERR, useSets, this); |
|
parser.reportError(Constants.ERROR, err); |
|
} |
|
_useSets = new UseAttributeSets(useSets, parser); |
|
} |
|
parseChildren(parser); |
|
} |
|
|
|
public void display(int indent) { |
|
indent(indent); |
|
Util.println("Copy"); |
|
indent(indent + IndentIncrement); |
|
displayContents(indent + IndentIncrement); |
|
} |
|
|
|
public Type typeCheck(SymbolTable stable) throws TypeCheckError { |
|
if (_useSets != null) { |
|
_useSets.typeCheck(stable); |
|
} |
|
typeCheckContents(stable); |
|
return Type.Void; |
|
} |
|
|
|
public void translate(ClassGenerator classGen, MethodGenerator methodGen) { |
|
final ConstantPoolGen cpg = classGen.getConstantPool(); |
|
final InstructionList il = methodGen.getInstructionList(); |
|
|
|
final LocalVariableGen name = |
|
methodGen.addLocalVariable2("name", |
|
Util.getJCRefType(STRING_SIG), |
|
null); |
|
final LocalVariableGen length = |
|
methodGen.addLocalVariable2("length", |
|
Util.getJCRefType("I"), |
|
null); |
|
|
|
|
|
il.append(methodGen.loadDOM()); |
|
il.append(methodGen.loadCurrentNode()); |
|
il.append(methodGen.loadHandler()); |
|
final int cpy = cpg.addInterfaceMethodref(DOM_INTF, |
|
"shallowCopy", |
|
"(" |
|
+ NODE_SIG |
|
+ TRANSLET_OUTPUT_SIG |
|
+ ")" + STRING_SIG); |
|
il.append(new INVOKEINTERFACE(cpy, 3)); |
|
il.append(DUP); |
|
name.setStart(il.append(new ASTORE(name.getIndex()))); |
|
final BranchHandle ifBlock1 = il.append(new IFNULL(null)); |
|
|
|
|
|
il.append(new ALOAD(name.getIndex())); |
|
final int lengthMethod = cpg.addMethodref(STRING_CLASS,"length","()I"); |
|
il.append(new INVOKEVIRTUAL(lengthMethod)); |
|
il.append(DUP); |
|
length.setStart(il.append(new ISTORE(length.getIndex()))); |
|
|
|
// Ignore attribute sets if current node is ROOT. DOM.shallowCopy() |
|
|
|
final BranchHandle ifBlock4 = il.append(new IFEQ(null)); |
|
|
|
|
|
if (_useSets != null) { |
|
// If the parent of this element will result in an element being |
|
|
|
final SyntaxTreeNode parent = getParent(); |
|
if ((parent instanceof LiteralElement) || |
|
(parent instanceof LiteralElement)) { |
|
_useSets.translate(classGen, methodGen); |
|
} |
|
// If not we have to check to see if the copy will result in an |
|
|
|
else { |
|
|
|
il.append(new ILOAD(length.getIndex())); |
|
final BranchHandle ifBlock2 = il.append(new IFEQ(null)); |
|
|
|
_useSets.translate(classGen, methodGen); |
|
|
|
ifBlock2.setTarget(il.append(NOP)); |
|
} |
|
} |
|
|
|
|
|
ifBlock4.setTarget(il.append(NOP)); |
|
translateContents(classGen, methodGen); |
|
|
|
// Call the output handler's endElement() if we copied an element |
|
|
|
length.setEnd(il.append(new ILOAD(length.getIndex()))); |
|
final BranchHandle ifBlock3 = il.append(new IFEQ(null)); |
|
il.append(methodGen.loadHandler()); |
|
name.setEnd(il.append(new ALOAD(name.getIndex()))); |
|
il.append(methodGen.endElement()); |
|
|
|
final InstructionHandle end = il.append(NOP); |
|
ifBlock1.setTarget(end); |
|
ifBlock3.setTarget(end); |
|
methodGen.removeLocalVariable(name); |
|
methodGen.removeLocalVariable(length); |
|
} |
|
} |