/* * SPDX-License-Identifier: Apache-2.0 * * The OpenSearch Contributors require contributions made to * this file be licensed under the Apache-2.0 license or a * compatible open source license. */ /* * Licensed to Elasticsearch under one or more contributor * license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright * ownership. Elasticsearch 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. */ /* * Modifications Copyright OpenSearch Contributors. See * GitHub history for details. */ package org.opensearch.painless.ir; import org.opensearch.painless.ClassWriter; import org.opensearch.painless.Location; import org.opensearch.painless.MethodWriter; import org.opensearch.painless.ScriptClassInfo; import org.opensearch.painless.phase.IRTreeVisitor; import org.opensearch.painless.symbol.ScriptScope; import org.opensearch.painless.symbol.WriteScope; import org.objectweb.asm.ClassVisitor; import org.objectweb.asm.Opcodes; import org.objectweb.asm.Type; import org.objectweb.asm.commons.Method; import org.objectweb.asm.util.Printer; import java.lang.invoke.MethodType; import java.util.ArrayList; import java.util.BitSet; import java.util.List; import static org.opensearch.painless.WriterConstants.BASE_INTERFACE_TYPE; import static org.opensearch.painless.WriterConstants.CLASS_TYPE; public class ClassNode extends IRNode { /* ---- begin tree structure ---- */ private final List fieldNodes = new ArrayList<>(); private final List functionNodes = new ArrayList<>(); private final BlockNode clinitBlockNode; public void addFieldNode(FieldNode fieldNode) { fieldNodes.add(fieldNode); } public List getFieldsNodes() { return fieldNodes; } public void addFunctionNode(FunctionNode functionNode) { functionNodes.add(functionNode); } public List getFunctionsNodes() { return functionNodes; } public BlockNode getClinitBlockNode() { return clinitBlockNode; } /* ---- end tree structure, begin node data ---- */ private Printer debugStream; private ScriptScope scriptScope; public void setDebugStream(Printer debugStream) { this.debugStream = debugStream; } public Printer getDebugStream() { return debugStream; } public void setScriptScope(ScriptScope scriptScope) { this.scriptScope = scriptScope; } public ScriptScope getScriptScope() { return scriptScope; } /* ---- end node data, begin visitor ---- */ @Override public void visit(IRTreeVisitor irTreeVisitor, Scope scope) { irTreeVisitor.visitClass(this, scope); } @Override public void visitChildren(IRTreeVisitor irTreeVisitor, Scope scope) { clinitBlockNode.visit(irTreeVisitor, scope); for (FunctionNode functionNode : functionNodes) { functionNode.visit(irTreeVisitor, scope); } } /* ---- end visitor ---- */ public ClassNode(Location location) { super(location); clinitBlockNode = new BlockNode(new Location("internal$clinit$blocknode", 0)); clinitBlockNode.setAllEscape(true); } public byte[] write() { ScriptClassInfo scriptClassInfo = scriptScope.getScriptClassInfo(); BitSet statements = new BitSet(scriptScope.getScriptSource().length()); scriptScope.addStaticConstant("$STATEMENTS", statements); // Create the ClassWriter. int classFrames = org.objectweb.asm.ClassWriter.COMPUTE_FRAMES | org.objectweb.asm.ClassWriter.COMPUTE_MAXS; int classAccess = Opcodes.ACC_PUBLIC | Opcodes.ACC_SUPER | Opcodes.ACC_FINAL; String interfaceBase = BASE_INTERFACE_TYPE.getInternalName(); String className = CLASS_TYPE.getInternalName(); String[] classInterfaces = new String[] { interfaceBase }; ClassWriter classWriter = new ClassWriter( scriptScope.getCompilerSettings(), statements, debugStream, scriptClassInfo.getBaseClass(), classFrames, classAccess, className, classInterfaces ); ClassVisitor classVisitor = classWriter.getClassVisitor(); classVisitor.visitSource(Location.computeSourceName(scriptScope.getScriptName()), null); org.objectweb.asm.commons.Method init; if (scriptClassInfo.getBaseClass().getConstructors().length == 0) { init = new org.objectweb.asm.commons.Method("", MethodType.methodType(void.class).toMethodDescriptorString()); } else { init = new org.objectweb.asm.commons.Method( "", MethodType.methodType(void.class, scriptClassInfo.getBaseClass().getConstructors()[0].getParameterTypes()) .toMethodDescriptorString() ); } // Write the constructor: MethodWriter constructor = classWriter.newMethodWriter(Opcodes.ACC_PUBLIC, init); constructor.visitCode(); constructor.loadThis(); constructor.loadArgs(); constructor.invokeConstructor(Type.getType(scriptClassInfo.getBaseClass()), init); constructor.returnValue(); constructor.endMethod(); if (clinitBlockNode.getStatementsNodes().isEmpty() == false) { MethodWriter methodWriter = classWriter.newMethodWriter( Opcodes.ACC_PUBLIC | Opcodes.ACC_STATIC, new Method("", Type.getType(void.class), new Type[0]) ); clinitBlockNode.write(classWriter, methodWriter, new WriteScope()); methodWriter.returnValue(); methodWriter.endMethod(); } // Write all fields: for (FieldNode fieldNode : fieldNodes) { fieldNode.write(classWriter, null, null); } // Write all functions: for (FunctionNode functionNode : functionNodes) { functionNode.write(classWriter, null, new WriteScope()); } // End writing the class and store the generated bytes. classVisitor.visitEnd(); return classWriter.getClassBytes(); } @Override public void write(ClassWriter classWriter, MethodWriter methodWriter, WriteScope writeScope) { throw new UnsupportedOperationException("use write() instead"); } }