Package de.maramuse.soundcomp.math

Source Code of de.maramuse.soundcomp.math.Variable

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;
  }
}
TOP

Related Classes of de.maramuse.soundcomp.math.Variable

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.