Source

orange / source / orange / root.cpp

Full commit
/*
    This file is part of Orange.
    
    Copyright 1996-2010 Faculty of Computer and Information Science, University of Ljubljana
    Contact: janez.demsar@fri.uni-lj.si

    Orange is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    Orange 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 General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with Orange.  If not, see <http://www.gnu.org/licenses/>.
*/


#include "values.hpp"
#include "errors.hpp"

#include "root.ppp"


TOrange::TWarningFunction *TOrange::warningFunction = NULL;

TPropertyDescription _no_properties[] = { {NULL} };

size_t const _no_components[] = { 0 };

bool castableTo(const TClassDescription *objecttype, const TClassDescription *basetype)
{ do
    if (objecttype==basetype)
      return true;
    else
      objecttype = objecttype->base;
  while (objecttype);

  return false;
}




TOrange::~TOrange()
{}


void TOrange::afterSet(const char *)
{}


const type_info &TOrange::propertyType(const char *name) const
{ return *propertyDescription(name)->type; }


bool TOrange::hasProperty(const char *name) const
{ for (const TPropertyDescription *pd = classDescription()->properties; pd->name; pd++)
    if (!strcmp(pd->name, name))
      return true;
  return false;
}

const TPropertyDescription *TOrange::propertyDescription(const char *name, bool noException) const
{ for (const TPropertyDescription *pd = classDescription()->properties; pd->name; pd++)
    if (!strcmp(pd->name, name))
      return pd;

  if (!noException)
    raiseError("there is no property '%s'", name);

  return NULL;
}


void TOrange::getProperty_TValue(const TPropertyDescription *pd, TValue &b) const
{ b = *(TValue const *)CONST_MEMBER(pd->offset); }

#define MEMBER(ofs) (((char *)(this)) + ofs)

#define SIMPLE_GETSET_PROPERTY(_TYPE)                           \
void TOrange::setProperty(const char *name, const _TYPE &b)     \
{ const TPropertyDescription *pd = propertyDescription(name);   \
  if (pd->readOnly)                                             \
    raiseError("'%s.%s' is read-only", TYPENAME(typeid(*this)), name);             \
  if (*pd->type != typeid(_TYPE))                               \
    raiseError("type mismatch, unable to set '%s.%s'", TYPENAME(typeid(*this)), name);           \
  *(_TYPE *)MEMBER(pd->offset) = b;                             \
  afterSet(name);                                               \
}                                                               \
                                                                \
                                                                \
void TOrange::getProperty(const char *name, _TYPE &b) const     \
{ const TPropertyDescription *pd = propertyDescription(name);   \
  if (*pd->type != typeid(_TYPE))                               \
    raiseError("type mismatch, unable to read '%s.%s'", TYPENAME(typeid(*this)), name);         \
  b = *(_TYPE const *)CONST_MEMBER(pd->offset);                 \
}                                                               \
                                                                \



SIMPLE_GETSET_PROPERTY(bool)
SIMPLE_GETSET_PROPERTY(int)
SIMPLE_GETSET_PROPERTY(float)
SIMPLE_GETSET_PROPERTY(string)
SIMPLE_GETSET_PROPERTY(TValue)

#undef SIMPLE_GETSET_PROPERTY


#include "examples.hpp"

void TOrange::wr_setProperty(const char *name, const POrange &b)
{ const TPropertyDescription *pd = propertyDescription(name);
  if (pd->readOnly)
    raiseError("'%s.%s' is read-only", TYPENAME(typeid(*this)), name);
  if (pd->type == &typeid(TExample)) {
    if (b && !b.is_derived_from(TExample))
      raiseError("type mismatch, unable to set '%s.%s' (expected 'orange.Example', got %s).", TYPENAME(typeid(*this)), name, TYPENAME(typeid(b)));
  }
  else {
    if (b && !castableTo(b->classDescription(), pd->classDescription))
      raiseError("type mismatch, unable to set '%s.%s' (expected %s, got %s).", TYPENAME(typeid(*this)), name, TYPENAME(*pd->type), TYPENAME(typeid(b)));
  }

  *(POrange *)MEMBER(pd->offset) = b;
  afterSet(name);
}

  
void TOrange::wr_getProperty(const char *name, POrange &b) const
{ const TPropertyDescription *pd = propertyDescription(name);
  // Warning: no type checking here. We cannot guess what the user wants the pointer b to point to.
  b = *(POrange const *)CONST_MEMBER(pd->offset);
}


int TOrange::traverse(visitproc visit, void *arg) const
{ for(size_t const *ci = classDescription()->components; *ci; ci++) {
    // shouldn't take POrange's -- don't want to create any additional references
    TGCCounter *ptr = ((POrange const *)CONST_MEMBER(*ci))->counter;
    if (ptr)
      VISIT(ptr)
  }
  return 0;
}


int TOrange::dropReferences()
{ for(size_t const *ci = classDescription()->components; *ci; ci++)
    // Strictly speaking, this is wrong -- we assign POrange to a field of type, say, PVariable.
    // However, POrange(NULL) doesn't binary differ from PVariable(NULL)
    // And even if it did, the state of the object after dropReference doesn't matter;
    // what matters is getting rid of all members. 
    *(POrange *)MEMBER(*ci) = POrange();
  return 0;
}



extern char excbuf[512], excbuf2[512]; // defined in errors.cpp

void TOrange::raiseError(const char *anerr, ...) const
{ va_list vargs;
  #ifdef HAVE_STDARG_PROTOTYPES
    va_start(vargs, anerr);
  #else
    va_start(vargs);
  #endif

  snprintf(excbuf, 512, "'orange.%s': %s", TYPENAME(typeid(*this)), anerr);
  vsnprintf(excbuf2, 512, excbuf, vargs);
  throw mlexception(excbuf2);
}


void TOrange::raiseErrorWho(const char *who, const char *anerr, ...) const
{ va_list vargs;
  #ifdef HAVE_STDARG_PROTOTYPES
    va_start(vargs, anerr);
  #else
    va_start(vargs);
  #endif

  snprintf(excbuf, 512, "'orange.%s.%s': %s", TYPENAME(typeid(*this)), who, anerr);
  vsnprintf(excbuf2, 512, excbuf, vargs);
  throw mlexception(excbuf2);
}


void TOrange::raiseWarning(const char *anerr, ...) const
{ 
  if (warningFunction) {
    va_list vargs;
    #ifdef HAVE_STDARG_PROTOTYPES
      va_start(vargs, anerr);
    #else
      va_start(vargs);
    #endif

    snprintf(excbuf, 512, "'%s': %s", TYPENAME(typeid(*this)), anerr);
    vsnprintf(excbuf2, 512, excbuf, vargs);
    warningFunction(false, excbuf2);
  }
}

void TOrange::raiseCompatibilityWarning(const char *anerr, ...) const
{ 
  if (warningFunction) {
    va_list vargs;
    #ifdef HAVE_STDARG_PROTOTYPES
      va_start(vargs, anerr);
    #else
      va_start(vargs);
    #endif

    snprintf(excbuf, 512, "'%s': %s", TYPENAME(typeid(*this)), anerr);
    vsnprintf(excbuf2, 512, excbuf, vargs);
    warningFunction(true, excbuf2);
  }
}

void TOrange::raiseWarningWho(const char *who, const char *anerr, ...) const
{ 
  if (warningFunction) {
    va_list vargs;
    #ifdef HAVE_STDARG_PROTOTYPES
      va_start(vargs, anerr);
    #else
      va_start(vargs);
    #endif

    snprintf(excbuf, 512, "'orange.%s.%s': %s", TYPENAME(typeid(*this)), who, anerr);
    vsnprintf(excbuf2, 512, excbuf, vargs);
    warningFunction(false, excbuf2);
  }
}