package de.maramuse.soundcomp.math;
/*
* Copyright 2010 Jan Schmidt-Reinisch
*
* SoundComp - a sound processing library
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; in version 2.1
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
/*
* This class implements a "stateless variable". On each cycle, it is calculated at most once,
* on the first query of the value which is then cached until the variable is marked dirty again.
*
* This variable type can be used in all kinds of processes, to abbreviate calculations
* in case more than one process element should depend on the same calculated value.
*/
import de.maramuse.soundcomp.process.NamedSource;
import de.maramuse.soundcomp.process.ParameterMap;
import de.maramuse.soundcomp.process.ProcessElement;
import de.maramuse.soundcomp.process.SourceStore;
import de.maramuse.soundcomp.process.StandardParameters;
import de.maramuse.soundcomp.process.TypeMismatchException;
import de.maramuse.soundcomp.process.UnknownConnectionException;
import de.maramuse.soundcomp.process.UnnamedSource;
import de.maramuse.soundcomp.process.ValueType;
import de.maramuse.soundcomp.process.StandardParameters.Parameter;
import de.maramuse.soundcomp.util.NativeObjects;
import de.maramuse.soundcomp.util.ReadOnlyMap;
import de.maramuse.soundcomp.util.ReadOnlyMapImpl;
public class Variable implements ProcessElement, UnnamedSource {
// a variable is always filled the first time it is read in each cycle
// on the next cycle, it has to be marked dirty before evaluating the network
private long nativeSpace;
private double value;
private String abstractName;
private String instanceName;
private boolean dirty;
private SourceStore source;
private boolean evaluating=false;
public Variable() {
NativeObjects.registerNativeObject(this);
}
private static final ReadOnlyMapImpl<Integer, ValueType> destMap=new ReadOnlyMapImpl<Integer, ValueType>();
private static final ReadOnlyMapImpl<Integer, ValueType> srcMap=new ReadOnlyMapImpl<Integer, ValueType>();
private final ReadOnlyMapImpl<Integer, SourceStore> sourceMap=new ReadOnlyMapImpl<Integer, SourceStore>();
private SourceStore src;
private static ParameterMap outputsMap=new ParameterMap();
private static ParameterMap inputsMap=new ParameterMap();
static{
outputsMap.put(StandardParameters.OUT);
}
static{
destMap.put(StandardParameters.IN.i, ValueType.STREAM);
srcMap.put(StandardParameters.OUT.i, ValueType.STREAM);
}
@Override
public double getValue(int index) {
return getValue();
}
@Override
public void advanceState() {
// nothing to do. the value is calculated on first use in each cycle. It must be done this way
// so the value is guaranteed to be there when it is needed. The order in which advanceState of
// the various objects is called is unspecified, and so it would happen that some formula needs
// a variable that has not yet been calculated within the same sample cycle.
}
@Override
public void advanceOutput() {
// almost nothing to do. variables are stateless in the SoundComp meaning of the word. only mark the value as dirty.
dirty=true;
}
@Override
public void setAbstractName(String abstractName) {
this.abstractName=abstractName;
}
@Override
public String getAbstractName() {
return abstractName;
}
@Override
public void setInstanceName(String instanceName) {
this.instanceName=instanceName;
}
@Override
public String getInstanceName() {
return instanceName;
}
@Override
public long getNativeSpace() {
return nativeSpace;
}
@Override
public ReadOnlyMap<Integer, ValueType> getSourceTypes() {
return srcMap;
}
@Override
public ReadOnlyMap<String, Parameter> outputsByName() {
return outputsMap;
}
public void markDirty() {
dirty=true;
}
public void markClean() {
dirty=false;
}
@Override
public Variable clone(){
Variable v=new Variable();
v.abstractName=abstractName;
return v;
}
@Override
public ReadOnlyMap<Integer, ValueType> getDestinationTypes() {
return destMap;
}
@Override
public void setSource(int connectionIndex, NamedSource source, int sourceIndex)
throws UnknownConnectionException, TypeMismatchException {
src=new SourceStore(source, sourceIndex);
sourceMap.clear();
sourceMap.put(StandardParameters.OUT.i, src);
}
@Override
public ReadOnlyMap<Integer, SourceStore> getSourceMap() {
return sourceMap;
}
@Override
public ReadOnlyMap<String, Parameter> inputsByName() {
return inputsMap;
}
@Override
public ValueType getSourceType() {
return ValueType.STREAM;
}
@Override
public double getValue() {
if(evaluating)
throw new IllegalStateException("evaluation of variable "+abstractName+" leads to recursive calculation");
if(dirty){
evaluating=true;
value=source.getValue();
evaluating=false;
dirty=false;
}
return value;
}
}