354 lines
13 KiB
C++
354 lines
13 KiB
C++
|
/*
|
||
|
* Open BEAGLE
|
||
|
* Copyright (C) 2001-2007 by Christian Gagne and Marc Parizeau
|
||
|
*
|
||
|
* 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; either
|
||
|
* version 2.1 of the License, or (at your option) any later version.
|
||
|
*
|
||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||
|
*
|
||
|
* Contact:
|
||
|
* Laboratoire de Vision et Systemes Numeriques
|
||
|
* Departement de genie electrique et de genie informatique
|
||
|
* Universite Laval, Quebec, Canada, G1K 7P4
|
||
|
* http://vision.gel.ulaval.ca
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/*!
|
||
|
* \file beagle/GP/src/Primitive.cpp
|
||
|
* \brief Implementation of the type GP::Primitive.
|
||
|
* \author Christian Gagne
|
||
|
* \author Marc Parizeau
|
||
|
* \author Matthew Walker
|
||
|
* $Revision: 1.21.2.2 $
|
||
|
* $Date: 2007/09/10 18:24:11 $
|
||
|
*/
|
||
|
|
||
|
#include "beagle/GP.hpp"
|
||
|
|
||
|
using namespace Beagle;
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Construct a primitive.
|
||
|
* \param inNumberArguments Number of arguments of the primitive.
|
||
|
* \param inName Name of the primitive.
|
||
|
*/
|
||
|
GP::Primitive::Primitive(unsigned int inNumberArguments, Beagle::string inName) :
|
||
|
NamedObject(inName),
|
||
|
mNumberArguments(inNumberArguments)
|
||
|
{ }
|
||
|
|
||
|
|
||
|
#ifdef BEAGLE_HAVE_RTTI
|
||
|
|
||
|
/*!
|
||
|
* \brief Return the nth argument requested return type (for STGP).
|
||
|
* \param inN Index of the argument to get the type.
|
||
|
* \param ioContext Evolutionary context.
|
||
|
* \return Type_info (RTTI) with the argument's node type.
|
||
|
* \throw Beagle::InternalException If the method is not overdefined is a subclass.
|
||
|
*/
|
||
|
const std::type_info* GP::Primitive::getArgType(unsigned int inN, GP::Context& ioContext) const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
throw Beagle::InternalException(string("Method \"getArgType\" of class \"GP::Primitive\"").
|
||
|
append(" is called but is not properly overdefined in primitive \"").append(getName()).
|
||
|
append("\""));
|
||
|
Beagle_StackTraceEndM("const std::type_info* GP::Primitive::getArgType(unsigned int inN, GP::Context& ioContext) const");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Return the primitive return type (for STGP).
|
||
|
* \param inDesiredType Desired return type.
|
||
|
* \param inNumberArguments Number of arguments to get the return type from.
|
||
|
* \param ioContext Evolutionary context.
|
||
|
* \return Type_info (RTTI) with the primitive return type.
|
||
|
* \throw Beagle::InternalException If the method is not overdefined is a subclass.
|
||
|
*/
|
||
|
const std::type_info* GP::Primitive::getReturnType(GP::Context& ioContext) const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
throw Beagle::InternalException(string("Method \"getReturnType\" of class \"GP::Primitive\"").
|
||
|
append(" is called but is not properly overdefined in primitive \"").append(getName()).
|
||
|
append("\""));
|
||
|
Beagle_StackTraceEndM("const std::type_info* GP::Primitive::getReturnType(GP::Context& ioContext) const");
|
||
|
}
|
||
|
|
||
|
#endif // BEAGLE_HAVE_RTTI
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Return the children's node index.
|
||
|
* \param inN Index of the children as arguments (index 0 is first arguments).
|
||
|
* \param ioContext Actual evolutionary context.
|
||
|
* \return Index of the given nth children.
|
||
|
* \throw Beagle::AssertException If the children index is out-of-bound.
|
||
|
*/
|
||
|
unsigned int GP::Primitive::getChildrenNodeIndex(unsigned int inN, GP::Context& ioContext) const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
Beagle_UpperBoundCheckAssertM(inN,mNumberArguments-1);
|
||
|
GP::Tree& lActualTree = ioContext.getGenotype();
|
||
|
unsigned int lNodeIndex = ioContext.getCallStackTop() + 1;
|
||
|
for(unsigned int i=0; i<inN; i++) lNodeIndex += lActualTree[lNodeIndex].mSubTreeSize;
|
||
|
return lNodeIndex;
|
||
|
Beagle_StackTraceEndM("unsigned int GP::Primitive::getChildrenNodeIndex(unsigned int inN, GP::Context& ioContext) const");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Get the selection weight multiplier for the primitive.
|
||
|
* \param inNumberArguments Number of arguments for the selection weight.
|
||
|
* \param ioContext Evolutionary context.
|
||
|
* \return Selection weight multiplier.
|
||
|
*/
|
||
|
double GP::Primitive::getSelectionWeight(unsigned int inNumberArguments, GP::Context& ioContext) const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
if(inNumberArguments == GP::Primitive::eAny) return 1.0;
|
||
|
else if(inNumberArguments == GP::Primitive::eBranch) {
|
||
|
if(mNumberArguments > 0) return 1.0;
|
||
|
else return 0.0;
|
||
|
}
|
||
|
else if(inNumberArguments == mNumberArguments) return 1.0;
|
||
|
return 0.0;
|
||
|
Beagle_StackTraceEndM("double GP::Primitive::getSelectionWeight(unsigned int inNumberArguments, GP::Context& ioContext) const");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Get the primitive value.
|
||
|
* \param outValue Object in which the actual primitive value is written.
|
||
|
* \throw Beagle::ObjectException If the method is not overdefined is a subclass.
|
||
|
*/
|
||
|
void GP::Primitive::getValue(Object& outValue)
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
throw Beagle::InternalException(string("Method \"getValue\" of class \"GP::Primitive\"").
|
||
|
append(" is called but is not properly overdefined in primitive \"").append(getName()).
|
||
|
append("\""));
|
||
|
Beagle_StackTraceEndM("void GP::Primitive::getValue(Object& outValue)");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Give a reference to the current primitive.
|
||
|
* \param inNumberArguments Number of arguments asked for.
|
||
|
* \param ioContext Reference to the evolution context.
|
||
|
* \return Handle to the refered primitive.
|
||
|
*/
|
||
|
GP::Primitive::Handle GP::Primitive::giveReference(unsigned int inNumberArguments,
|
||
|
GP::Context& ioContext)
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
return GP::Primitive::Handle(this);
|
||
|
Beagle_StackTraceEndM("GP::Primitive::Handle GP::Primitive::giveReference(unsigned int inNumberArguments, GP::Context& ioContext)");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Return whether primitive's getValue and setValue are effective.
|
||
|
* \return True if getValue/setValue method are effective, false if not.
|
||
|
*/
|
||
|
bool GP::Primitive::haveValue() const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
return false;
|
||
|
Beagle_StackTraceEndM("bool GP::Primitive::haveValue() const");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Initialize the primitive.
|
||
|
* \param ioSystem Evolutionary system used.
|
||
|
*/
|
||
|
void GP::Primitive::initialize(GP::System& ioSystem)
|
||
|
{ }
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Compare equality of two primitives. Name, number of args. and attribute must be the same.
|
||
|
* \param inRightObj Primitive to compare to the actual one.
|
||
|
* \return True if primitive are identical, false if not.
|
||
|
*/
|
||
|
bool GP::Primitive::isEqual(const Object& inRightObj) const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
const GP::Primitive& lRightPrimitive = castObjectT<const GP::Primitive&>(inRightObj);
|
||
|
if((getName() != lRightPrimitive.getName()) ||
|
||
|
(mNumberArguments != lRightPrimitive.mNumberArguments)) return false;
|
||
|
return true;
|
||
|
Beagle_StackTraceEndM("bool GP::Primitive::isEqual(const Object& inRightObj) const");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Stability of selection weight. A weight is said stable if it doesn't change over time.
|
||
|
* \param inNumberArguments Number of arguments to check stability of selection weight.
|
||
|
* \return True if selection weight is stable, false if not.
|
||
|
*/
|
||
|
bool GP::Primitive::isSelectionWeightStable(unsigned int inNumberArguments) const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
return true;
|
||
|
Beagle_StackTraceEndM("bool GP::Primitive::isSelectionWeightStable(unsigned int inNumberArguments) const");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Read the primitive from XML subtree.
|
||
|
* \param inIter XML iterator used to read the primitive from.
|
||
|
* \param ioContext Evolutionary context.
|
||
|
* \throw IOException If primitive format is not respected.
|
||
|
*/
|
||
|
void GP::Primitive::readWithContext(PACC::XML::ConstIterator inIter, GP::Context& ioContext)
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
if(inIter->getType() != PACC::XML::eData)
|
||
|
throw Beagle_IOExceptionNodeM(*inIter, "tag expected!");
|
||
|
|
||
|
if(inIter->getValue() != getName().c_str()) {
|
||
|
std::ostringstream lOSS;
|
||
|
lOSS << "tag <" << getName() << "> expected, but ";
|
||
|
lOSS << "got tag <" << inIter->getValue() << "> instead!";
|
||
|
throw Beagle_IOExceptionNodeM(*inIter, lOSS.str().c_str());
|
||
|
}
|
||
|
Beagle_StackTraceEndM("void GP::Primitive::readWithContext(PACC::XML::ConstIterator inIter, GP::Context& ioContext)");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Set the value of a primitive.
|
||
|
* \param inValue Value of the primitive.
|
||
|
* \throw Beagle::ObjectException If the method is not overdefined is a subclass.
|
||
|
*/
|
||
|
void GP::Primitive::setValue(const Object& inValue)
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
throw Beagle::InternalException(string("Method \"setValue\" of class \"GP::Primitive\"").
|
||
|
append(" is called but is not properly overdefined in primitive \"").append(getName()).
|
||
|
append("\""));
|
||
|
Beagle_StackTraceEndM("void GP::Primitive::setValue(const Object& inValue)");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Validate the primitive connections in the tree when using contrained or strongly-typed GP.
|
||
|
* \param ioContext Evolutionary context.
|
||
|
* \return True if the tree topology is correct, false if not.
|
||
|
* \throw Beagle::AssertException If the context is in a bad state.
|
||
|
*/
|
||
|
bool GP::Primitive::validate(GP::Context& ioContext) const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
#ifdef BEAGLE_HAVE_RTTI
|
||
|
|
||
|
Beagle_AssertM(ioContext.getCallStackSize() != 0);
|
||
|
GP::Tree& lActualTree = ioContext.getGenotype();
|
||
|
|
||
|
// If the primitive is the root of the tree, check its type is correct.
|
||
|
if(ioContext.getCallStackTop()==0) {
|
||
|
const std::type_info* lTreeRootType = lActualTree.getRootType(ioContext);
|
||
|
const std::type_info* lReturnType = getReturnType(ioContext);
|
||
|
if((lTreeRootType==NULL) || ((lReturnType!=NULL) && ((*lTreeRootType)==(*lReturnType)))) {
|
||
|
// Valid.
|
||
|
return true;
|
||
|
}
|
||
|
else {
|
||
|
// Not valid.
|
||
|
Beagle_LogVerboseM(
|
||
|
ioContext.getSystem().getLogger(),
|
||
|
"primitive", "Beagle::GP::Primitive",
|
||
|
string("Validation of \"").append(getName()).append("\" primitive failed. ").
|
||
|
append("The primitive is at the root of the ").
|
||
|
append(uint2ordinal(ioContext.getGenotypeIndex()+1)).append(" tree and its type (\"").
|
||
|
append(lReturnType->name()).append("\") does not match the type required as tree root ").
|
||
|
append("(\"").append(lTreeRootType->name()).append("\").")
|
||
|
);
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
unsigned int lParentIndex = ioContext.getCallStackElement(ioContext.getCallStackSize()-2);
|
||
|
unsigned int lChildIndex = lParentIndex + 1;
|
||
|
unsigned int lArgsIndexChild = 0;
|
||
|
while(lChildIndex != ioContext.getCallStackTop()) {
|
||
|
lChildIndex += lActualTree[lChildIndex].mSubTreeSize;
|
||
|
Beagle_AssertM(lChildIndex < lActualTree.size());
|
||
|
lArgsIndexChild++;
|
||
|
Beagle_AssertM(lArgsIndexChild < lActualTree[lParentIndex].mPrimitive->getNumberArguments());
|
||
|
}
|
||
|
|
||
|
// Check that the return type of this node matches the type expected by this node's parent.
|
||
|
unsigned int lActualNodeIndex = ioContext.getCallStackTop();
|
||
|
ioContext.popCallStack();
|
||
|
const std::type_info* lDesiredType =
|
||
|
lActualTree[lParentIndex].mPrimitive->getArgType(lArgsIndexChild, ioContext);
|
||
|
ioContext.pushCallStack(lActualNodeIndex);
|
||
|
const std::type_info* lReturnedType = getReturnType(ioContext);
|
||
|
if((lDesiredType==NULL) || (lReturnedType==NULL) || ((*lDesiredType)==(*lReturnedType))) {
|
||
|
// Valid.
|
||
|
return true;
|
||
|
}
|
||
|
else {
|
||
|
// Not valid.
|
||
|
Beagle_LogVerboseM(
|
||
|
ioContext.getSystem().getLogger(),
|
||
|
"primitive", "Beagle::GP::Primitive",
|
||
|
string("Validation of \"").append(getName()).append("\" primitive failed.").
|
||
|
append(" The primitive is at the ").
|
||
|
append(uint2ordinal(ioContext.getCallStackTop()+1)).append(" node of the ").
|
||
|
append(uint2ordinal(ioContext.getGenotypeIndex()+1)).append(" tree and its type (\"").
|
||
|
append(lReturnedType->name()).append("\") does not match the desired type (\"").
|
||
|
append(lDesiredType->name()).append("\") required by its parent (\"").
|
||
|
append(lActualTree[lParentIndex].mPrimitive->getName()).append("\") at the ").
|
||
|
append(uint2ordinal(lParentIndex)).append(" node of the same tree.")
|
||
|
);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
#else // BEAGLE_HAVE_RTTI
|
||
|
|
||
|
return true;
|
||
|
|
||
|
#endif // BEAGLE_HAVE_RTTI
|
||
|
Beagle_StackTraceEndM("bool GP::Primitive::validate(GP::Context& ioContext) const");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Write the primitive.
|
||
|
* \param ioStreamer XML streamer to write the primitive into.
|
||
|
* \param inIndent Whether XML output should be indented.
|
||
|
*/
|
||
|
void GP::Primitive::write(PACC::XML::Streamer& ioStreamer, bool inIndent) const
|
||
|
{
|
||
|
Beagle_StackTraceBeginM();
|
||
|
ioStreamer.openTag(getName().c_str(), inIndent);
|
||
|
writeContent(ioStreamer, inIndent);
|
||
|
ioStreamer.closeTag();
|
||
|
Beagle_StackTraceEndM("void GP::Primitive::write(PACC::XML::Streamer& ioStreamer, bool inIndent) const");
|
||
|
}
|
||
|
|
||
|
|
||
|
/*!
|
||
|
* \brief Write the primitive content.
|
||
|
* \param ioStreamer XML streamer to write the primitive into.
|
||
|
* \param inIndent Whether XML output should be indented.
|
||
|
*/
|
||
|
void GP::Primitive::writeContent(PACC::XML::Streamer& ioStreamer, bool inIndent) const
|
||
|
{ }
|
||
|
|