/* * File LikelihoodCore.java * * Copyright (C) 2010 Remco Bouckaert remco@cs.auckland.ac.nz * * This file is part of BEAST2. * See the NOTICE file distributed with this work for additional * information regarding copyright ownership and licensing. * * BEAST 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; either version 2 * of the License, or (at your option) any later version. * * BEAST 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 BEAST; if not, write to the * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301 USA */ package beast.evolution.likelihood; /** * The likelihood core is the class that performs the calculations * in the peeling algorithm (see Felsenstein, Joseph (1981). * Evolutionary trees from DNA sequences: a maximum likelihood approach. * J Mol Evol 17 (6): 368-376.). It does this by calculating the partial * results for all sites, node by node. The order in which the nodes * are visited is controlled by the TreeLikelihood. T * <p/> * In order to reuse computations of previous likelihood calculations, * a current state, and a stored state are maintained. Again, the TreeLikelihood * controls when to update from current to stored and vice versa. So, in * LikelihoodCore implementations, duplicates need to be kept for all partials. * Also, a set of indices to indicate which of the data is stored state and which * is current state is commonly the most efficient way to sort out which is which. */ abstract public class LikelihoodCore { /** * reserve memory for partials, indices and other * data structures required by the core * */ abstract public void initialize(int nodeCount, int patternCount, int matrixCount, boolean integrateCategories, boolean useAmbiguities); /** * clean up after last likelihood calculation, if at all required * */ @Override abstract public void finalize() throws java.lang.Throwable; /** * reserve memory for partials for node with number nodeIndex * */ abstract public void createNodePartials(int nodeIndex); /** * indicate that the partials for node * nodeIndex is about the be changed, that is, that the stored * state for node nodeIndex cannot be reused * */ abstract public void setNodePartialsForUpdate(int nodeIndex); /** * assign values of partials for node with number nodeIndex * */ abstract public void setNodePartials(int nodeIndex, double[] partials); abstract public void getNodePartials(int nodeIndex, double[] partials); //abstract public void setCurrentNodePartials(int nodeIndex, double[] partials); /** reserve memory for states for node with number nodeIndex **/ //abstract public void createNodeStates(int nodeIndex); /** * assign values of states for node with number nodeIndex * */ abstract public void setNodeStates(int nodeIndex, int[] states); abstract public void getNodeStates(int nodeIndex, int[] states); /** * indicate that the probability transition matrix for node * nodeIndex is about the be changed, that is, that the stored * state for node nodeIndex cannot be reused * */ abstract public void setNodeMatrixForUpdate(int nodeIndex); /** * assign values of states for probability transition matrix for node with number nodeIndex * */ abstract public void setNodeMatrix(int nodeIndex, int matrixIndex, double[] matrix); abstract public void getNodeMatrix(int nodeIndex, int matrixIndex, double[] matrix); /** assign values of states for probability transition matrices * padded with 1s for dealing with unknown characters for node with number nodeIndex **/ // abstract public void setPaddedNodeMatrices(int nodeIndex, double[] matrix); /** * indicate that the topology of the tree chanced so the cache * data structures cannot be reused * */ public void setNodeStatesForUpdate(int nodeIndex) { } ; /** * flag to indicate whether scaling should be used in the * likelihood calculation. Scaling can help in dealing with * numeric issues (underflow). */ boolean m_bUseScaling = false; abstract public void setUseScaling(double scale); public boolean getUseScaling() { return m_bUseScaling; } /** * return the cumulative scaling effect. Should be zero if no scaling is used * */ abstract public double getLogScalingFactor(int patternIndex_); /** * Calculate partials for node node3, with children node1 and node2Index. * NB Depending on whether the child nodes contain states or partials, the * calculation differs-* */ abstract public void calculatePartials(int node1, int node2Index, int node3); //abstract public void calculatePartials(int node1, int node2Index, int node3, int[] matrixMap); /** * integrate partials over categories (if any). * */ abstract public void integratePartials(int nodeIndex, double[] proportions, double[] outPartials); /** * calculate log likelihoods at the root of the tree, * using frequencies as root node distribution. * outLogLikelihoods contains the resulting probabilities for each of * the patterns * */ abstract public void calculateLogLikelihoods(double[] partials, double[] frequencies, double[] outLogLikelihoods); public void processStack() { } abstract protected void calculateIntegratePartials(double[] inPartials, double[] proportions, double[] outPartials); // abstract public void calcRootPsuedoRootPartials(double[] frequencies, int nodeIndex, double [] pseudoPartials); // abstract public void calcNodePsuedoRootPartials(double[] inPseudoPartials, int nodeIndex, double [] outPseudoPartials); // abstract public void calcPsuedoRootPartials(double [] parentPseudoPartials, int nodeIndex, double [] pseudoPartials); // abstract void integratePartialsP(double [] inPartials, double [] proportions, double [] m_fRootPartials); // abstract void calculateLogLikelihoodsP(double[] partials,double[] outLogLikelihoods); /** * store current state * */ abstract public void store(); /** * reset current state to stored state, only used when switching from non-scaled to scaled or vice versa * */ abstract public void unstore(); /** * restore state * */ abstract public void restore(); // /** do internal diagnosics, and suggest an alternative core if appropriate **/ // abstract LikelihoodCore feelsGood(); }