org.mvel2.optimizers.dynamic.DynamicGetAccessor Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of mvel2 Show documentation
Show all versions of mvel2 Show documentation
A fork of MVEL by Axibase. Original version is hosted here: https://github.com/mvel/mvel.
MVEL is a powerful expression language for Java-based applications.
It provides a plethora of features and is suited for everything
from the smallest property binding and extraction, to full blown scripts.
/**
* MVEL 2.0
* Copyright (C) 2007 The Codehaus
* Mike Brock, Dhanji Prasanna, John Graham, Mark Proctor
*
* Licensed 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 org.mvel2.optimizers.dynamic;
import org.mvel2.ParserContext;
import org.mvel2.compiler.Accessor;
import org.mvel2.integration.VariableResolverFactory;
import org.mvel2.optimizers.AccessorOptimizer;
import org.mvel2.optimizers.OptimizationNotSupported;
import org.mvel2.optimizers.OptimizerFactory;
import static java.lang.System.currentTimeMillis;
public class DynamicGetAccessor implements DynamicAccessor {
private char[] expr;
private int start;
private int offset;
private long stamp;
private int type;
private int runcount;
private boolean opt = false;
private ParserContext pCtx;
private Accessor _safeAccessor;
private Accessor _accessor;
public DynamicGetAccessor(ParserContext pCtx, char[] expr, int start, int offset, int type, Accessor _accessor) {
this._safeAccessor = this._accessor = _accessor;
this.type = type;
this.expr = expr;
this.start = start;
this.offset = offset;
this.pCtx = pCtx;
stamp = currentTimeMillis();
}
public Object getValue(Object ctx, Object elCtx, VariableResolverFactory variableFactory) {
if (!opt) {
if (++runcount > DynamicOptimizer.tenuringThreshold) {
if ((currentTimeMillis() - stamp) < DynamicOptimizer.timeSpan) {
opt = true;
try{
return optimize(ctx, elCtx, variableFactory);
}
catch(OptimizationNotSupported ex){
// If optimization fails then, rather than fail evaluation, fallback to use safe reflective accessor
}
}
else {
runcount = 0;
stamp = currentTimeMillis();
}
}
}
return _accessor.getValue(ctx, elCtx, variableFactory);
}
public Object setValue(Object ctx, Object elCtx, VariableResolverFactory variableFactory, Object value) {
runcount++;
return _accessor.setValue(ctx, elCtx, variableFactory, value);
}
private Object optimize(Object ctx, Object elCtx, VariableResolverFactory variableResolverFactory) {
if (DynamicOptimizer.isOverloaded()) {
DynamicOptimizer.enforceTenureLimit();
}
AccessorOptimizer ao = OptimizerFactory.getAccessorCompiler("ASM");
switch (type) {
case DynamicOptimizer.REGULAR_ACCESSOR:
_accessor = ao.optimizeAccessor(pCtx, expr, start, offset, ctx, elCtx, variableResolverFactory, false, null);
return ao.getResultOptPass();
case DynamicOptimizer.OBJ_CREATION:
_accessor = ao.optimizeObjectCreation(pCtx, expr, start, offset, ctx, elCtx, variableResolverFactory);
return _accessor.getValue(ctx, elCtx, variableResolverFactory);
case DynamicOptimizer.COLLECTION:
_accessor = ao.optimizeCollection(pCtx, ctx, null, expr, start, offset, ctx, elCtx, variableResolverFactory);
return _accessor.getValue(ctx, elCtx, variableResolverFactory);
}
return null;
}
public void deoptimize() {
this._accessor = this._safeAccessor;
opt = false;
runcount = 0;
stamp = currentTimeMillis();
}
public long getStamp() {
return stamp;
}
public int getRuncount() {
return runcount;
}
public Class getKnownEgressType() {
return _safeAccessor.getKnownEgressType();
}
public Accessor getAccessor() {
return _accessor;
}
public Accessor getSafeAccessor() {
return _safeAccessor;
}
}