Sophie

Sophie

distrib > Mandriva > 2010.0 > i586 > media > contrib-release > by-pkgid > b818b6f4a71f6d777a55341c7d17486c > files > 76

libgpsim-devel-0.22.0-1mdv2008.0.i586.rpm

/*
   Copyright (C) 1998 T. Scott Dattalo

This file is part of gpsim.

gpsim 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 2, or (at your option)
any later version.

gpsim 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 gpsim; see the file COPYING.  If not, write to
the Free Software Foundation, 59 Temple Place - Suite 330,
Boston, MA 02111-1307, USA.  */

//
// symbol.h
//

#include <string>
#include <vector>
#include "value.h"
#include "registers.h"
#include "exports.h"

using namespace std;

#ifndef __SYMBOL_H__
#define __SYMBOL_H__



class stimulus;
class Stimulus_Node;
class WREG;
class PortRegister;
class Processor;
class Register;
class Module;
class Expression;
class symbol;
class register_symbol;
class node_symbol;
class stimulus_symbol;
class module_symbol;
class attribute_symbol;

void display_symbol_file_error(int);


enum SYMBOL_TYPE
{
  SYMBOL_INVALID,
  SYMBOL_BASE_CLASS,
  SYMBOL_IOPORT,
  SYMBOL_STIMULUS_NODE,
  SYMBOL_STIMULUS,
  SYMBOL_LINE_NUMBER,
  SYMBOL_CONSTANT,
  SYMBOL_REGISTER,
  SYMBOL_ADDRESS,
  SYMBOL_SPECIAL_REGISTER,   // like W
  SYMBOL_PROCESSOR,
  SYMBOL_MODULE
};


class Symbol_Table : protected vector<Value*> {
private:
  struct NameLessThan : binary_function<Value*, Value*, bool> {
    bool operator()(const Value* left, const Value* right) const {
      return left->name() < right->name();
    }
  };
  iterator FindIt(const char *s);
  iterator FindIt(const string &s);
  iterator FindIt(Value *key);

public:
  Symbol_Table();
#ifndef _MSC_VER
   typedef vector<Value*> _Myt;
#endif
  typedef _Myt::iterator iterator;

  bool add(Value*);

  void add_ioport(PortRegister *ioport);
  void add_stimulus_node(Stimulus_Node *stimulus_node);
  void add_stimulus(stimulus *s);
  void add_line_number(int address, const char *symbol_name=0);
  void add_constant(const char *, int, bool bClearable = true);
  register_symbol* add_register(Register *reg, const char *symbol_name=0);
  register_symbol* add_register(Register *new_reg, const char *symbol_name,
                                unsigned int uMask );
  void add_address(const char *, int );
  void add_w(WREG *w );
  void add_module(Module * m, const char *module_name);

  void remove_module(Module * m);
  Value *remove(string &);
  void rename(const char *pOldName, const char *pNewName);
  void dump_all(void);
  void dump_one(const char *s);
  void dump_one(string *s);
  void dump_type(type_info const&t);
  void dump_filtered(const string & sSymbol);
  string DisplayType(type_info const &symt);

  Value *find(const char *s);
  Value *find(const string &s);
  Value *find(type_info const&t, const char *s);
  Register * findRegister(unsigned int address);
  Register * findRegister(const char *s);
  register_symbol * findRegisterSymbol(const char *pName);
  register_symbol * findRegisterSymbol(unsigned int uAddress);
  register_symbol * findRegisterSymbol(unsigned int uAddress,
                                       unsigned int uBitmask);
  const char *  findProgramAddressLabel(unsigned int address);

  // This was intended to be used by disassembly code. I (JR) added
  // this then I figured out that it is not useful unless
  // you have a cross reference to every address that references
  // the symbol. So I'm leaving the code for the future when
  // the constant object includes a list of address references.
  const char *  findConstant(unsigned int uValue,
    unsigned int uReferencedFromAddress);
  // Convenience functions for accessing symbols of a specific type.
  // In general, provide a string containing the symbol name to access
  // the symbol.
  Integer *     findInteger(const char *s);
  Boolean *     findBoolean(const char *s);
  module_symbol *findModuleSymbol(const char *s);
  Module *      findModule(const char *s);
  String *      findString(const char *s);

protected:
  // 'findSymbol' is a template that will create type-specific code
  // for searching the symbol table.
  template<class _symbol_t>
  _symbol_t * findSymbol(const char *s, _symbol_t*)
  {
    iterator sti = FindIt(s);
    while( sti != end()) {
      _symbol_t *pNode = dynamic_cast<_symbol_t*>(*sti);
      if(pNode != NULL) {
        int iResult = pNode->name().compare(s);
        if(iResult == 0) {
          return pNode;
        }
        else if(iResult > 0) {
          // leave early for efficiency
          return NULL;
        }
      }
      sti++;
    }
    return NULL;
  }

public:
  node_symbol *             findNodeSymbol(const char *s);
  Stimulus_Node *           findNode(const char *s);
  inline node_symbol *      findNodeSymbol(string &s) {
    return findNodeSymbol(s.c_str());
  }
  inline Stimulus_Node *    findNode(string &s) {
    return findNode(s.c_str());
  }

  stimulus_symbol *         findStimulusSymbol(const char *s);
  stimulus *                findStimulus(const char *s);
  inline stimulus_symbol *  findStimulusSymbol(string &s) {
    return findStimulusSymbol(s.c_str());
  }
  inline stimulus *         findStimulus(const string &s) {
    return findStimulus(s.c_str());
  }
  attribute_symbol *         findAttributeSymbol(const char *s);
  inline attribute_symbol *  findAttributeSymbol(string &s) {
    return findAttributeSymbol(s.c_str());
  }

  bool  Exist(const char *);
  void clear();
  void clear_all();
  void Initialize();
  void Reinitialize();

  iterator begin() {
    return _Myt::begin();
  }
  iterator end() {
    return _Myt::end();
  }

protected:
  template<class _symbol_t>
  class symbol_iterator_t : iterator {
  friend class module_symbol_iterator;
  public:
    symbol_iterator_t() {
      m_pSymbolTable = NULL;
    }

    symbol_iterator_t(const symbol_iterator_t & it) : iterator(it) {
      m_pSymbolTable = it.m_pSymbolTable;
    }

    symbol_iterator_t(Symbol_Table *pSymbolTable, iterator it) {
      *((iterator*)this) = it;
      m_pSymbolTable = pSymbolTable;
    }

    _symbol_t * operator*() const
			{	// return designated object
			return ((_symbol_t*)**(const_iterator *)this);
			}

    bool operator!=(const symbol_iterator_t& _Right) const {
      return (iterator)*this != (iterator) _Right;
    }

    symbol_iterator_t& operator++() {
      // preincrement
      iterator & it = (iterator&)*this;
      for(it++; it != m_pSymbolTable->end(); it++) {
        if(dynamic_cast<_symbol_t*>(*it) != NULL)
          return (*this);
      }
	    return (*this);
    }


    symbol_iterator_t operator++(int)
			{	// postincrement
			symbol_iterator_t _Tmp = *this;
			++*this;
			return (_Tmp);
			}
  protected:
    Symbol_Table *m_pSymbolTable;
  };

  ///
  /// Note:: Apparently every argument in the template also needs
  ///        to be in the function argument list. The function
  ///        arguments are only used as place holders to allow
  //         using this function template for a variety of functions
  ///        that only need to vary by return type.
  template<class _symbol_iterator_t, class _symbol_t>
  _symbol_iterator_t beginSymbol(_symbol_iterator_t *, _symbol_t*);
  template<class _symbol_iterator_t, class _symbol_t>
  _symbol_iterator_t endSymbol(_symbol_iterator_t *, _symbol_t*);


public:
  ///
  /// node_symbol iterator declarations
  ///

  typedef symbol_iterator_t<node_symbol> node_symbol_iterator;
  node_symbol_iterator beginNodeSymbol();
  node_symbol_iterator endNodeSymbol();

  ///
  /// stimulus_symbol iterator declarations
  ///
  typedef symbol_iterator_t<stimulus_symbol> stimulus_symbol_iterator;

  stimulus_symbol_iterator beginStimulusSymbol();
  stimulus_symbol_iterator endStimulusSymbol();

  ///
  /// module_symbol iterator declarations
  ///
  class module_symbol_iterator : public symbol_iterator_t<module_symbol> {
  public:
    module_symbol_iterator() {
    }

    module_symbol_iterator(const module_symbol_iterator & it) :
      symbol_iterator_t<module_symbol>(it) {
    }

    module_symbol_iterator(Symbol_Table *pSymbolTable, iterator it) :
      symbol_iterator_t<module_symbol>(pSymbolTable, it) {
    }

    module_symbol_iterator operator++(int);
  };
  module_symbol_iterator beginModuleSymbol();
  module_symbol_iterator endModuleSymbol();

  ///
  ///   Symbols defined from gpsim command line
  ///
  /*
  typedef list<char*> SymbolList;
  SymbolList s_CmdLineSymbolList;
  void AddFromCommandLine(char * pSymbol);
  void PopulateWithCommandLineSymbols();
  */
};



#if defined(_WIN32)
  #if !defined(IN_MODULE) 
  extern Symbol_Table symbol_table;
  #endif
// we are in Windows: don't access symbol_table object directly!
LIBGPSIM_EXPORT Symbol_Table & get_symbol_table(void);
#else
// we are in gpsim: use of get_symbol_table() is recommended,
// even if trace object can be accessed directly.
extern Symbol_Table symbol_table;
inline Symbol_Table &get_symbol_table(void) {
  return symbol_table;
}
#endif




//------------------------------------------------------------------------
/// symbol - base class for gpsim symbols. gpsim symbols are 'Value' objects
/// that are typically associated with a Typed object that has been defined
/// by a Module. Eg. a register_symbol is a symbol that is associtated with
/// a processor's registers.

class symbol : public Value
{
public:

  virtual Value* evaluate();
  virtual string toString();

  symbol(const char *);
  symbol(string &);
  virtual ~symbol();
};

class AliasedSymbol : public symbol {
public:
  AliasedSymbol(const char * pName, Value *pSymbol) :
    symbol(pName) {
    m_pData = pSymbol;
  }
  AliasedSymbol(const char * pName, Value *pSymbol, const char *pDoc) :
    symbol(pName) {
    m_pData = pSymbol;
    set_description(pDoc);
  }
  // The following functions are implemented to call the
  // contained object. The name() and new_name() function
  // will return and set the name of the alias.
  virtual void set(const char *v,int len=0) {m_pData->set(v, len);}
  virtual void set(double v) {m_pData->set(v);}
  virtual void set(gint64 v) {m_pData->set(v);}
  virtual void set(int v) {m_pData->set(v);}
  virtual void set(bool v) {m_pData->set(v);}
  virtual void set(Value *v) {m_pData->set(v);}
  virtual void set(Expression *v) {m_pData->set(v);}
  virtual void set(Packet &v) {m_pData->set(v);}

  /// Value 'get' methods provide a mechanism of casting Value objects
  /// to other value types. If the type cast is not supported in a
  /// derived class, an Error will be thrown.

  virtual void get(bool &v) {m_pData->get(v);}
  virtual void get(int &v) {m_pData->get(v);}
  virtual void get(guint64 &v) {m_pData->get(v);}
  virtual void get(gint64 &v) {m_pData->get(v);}
  virtual void get(double &v) {m_pData->get(v);}
  virtual void get(char *v, int len) {m_pData->get(v,len);}
  virtual void get(Packet &v) {m_pData->get(v);}
  virtual Value *getValue() { return m_pData; }

  virtual char *toString(char *v, int len) { return m_pData->toString(v,len);}
  virtual string toString() { return m_pData->toString(); }
  virtual Value* evaluate() { return m_pData->evaluate(); }
protected:
  Value *m_pData;
};

class node_symbol : public symbol
{
protected:
  Stimulus_Node *stimulus_node;
public:

  node_symbol(Stimulus_Node *);
  virtual string toString();
  Stimulus_Node * getNode() {
    return stimulus_node;
  }
};

class register_symbol : public symbol
{
protected:
  Register *reg;
  /** Scott to JR -- I don't understand this comment about m_uMask
    * m_uMask is used to mask out bits not associated
    * with the symbol when retrieving the value from the
    * register. Masked bit values are right shifted such
    * that if the 16th bit is high in the register and
    * m_uMask = 0x80, then the symbol value returned
    * or displayed will be 0x01.
    * Likewise to set the 16th bit of our example register_symbol
    * you would set the register_symbol value to 0x01 and
    * not 0x80. No other bits in the register are changed.
    * It works as if this symbol is its own register. The 
    * encoding into the target register is hidden by register_symbol.
    * The default value of m_uMask is 0xff so that no bits are
    * masked out.
    * A new version of Symbol_Table::add_register() may be used
    * to add masked register_symbol objects to the symbol table.
    */
  unsigned int m_uMask;
  unsigned int m_uMaskShift;

  unsigned int SetMaskedValue(unsigned int uValue);
  /// breakpoints
  virtual int set_break(ObjectBreakTypes bt=eBreakAny, ObjectActionTypes at=eActionHalt, Expression *expr=0);
  virtual int clear_break();

public:
  register_symbol(const register_symbol &);
  register_symbol(const char *, Register *);
  register_symbol(const char *, Register *, unsigned int uMask);
  register_symbol(Register *_reg);

  virtual symbol *copy();
  virtual string &name(void) const;
  virtual char *name(char *, int len);
  virtual string toString();

  virtual void get(int &);
  virtual void get(gint64 &);
  virtual void get(char *, int len);
  virtual void get(Packet &);

  virtual void set(int);
  virtual void set(Value *);
  virtual void set(const char *cP,int len=0);
  virtual void set(Packet &);

  virtual void update(void);

  inline operator gint64() {
    gint64 i;
    get(i);
    return i;
  }

  inline operator int() {
    gint64 i;
    get(i);
    return (int)i;
  }

  inline operator unsigned int() {
    gint64 i;
    get(i);
    return (unsigned int)i;
  }

  inline operator RegisterValue() {
    return getReg()->getRV_notrace();
  }

  inline register_symbol & operator =(const RegisterValue &i) {
    getReg()->putRV_notrace(i);
    return *this;
  }

  inline register_symbol & operator =(int i) {
    set(i);
    return *this;
  }

  inline register_symbol & operator =(unsigned int i) {
    set((int)i);
    return *this;
  }

  void setMask(Register *pReg);
  unsigned int getAddress(void);
  unsigned int getBitmask(void);

  virtual bool compare(ComparisonOperator *compOp, Value *rvalue);
  Value*       evaluate();

  Register *getReg();
  /// copy the object value to a user char array
  virtual char *toString(char *, int len);
  virtual char *toBitStr(char *, int len);

};

class ioport_symbol : public register_symbol
{
public:
  ioport_symbol(PortRegister *);
  ioport_symbol(Module *pParent, PortRegister *);
  PortRegister *getIOPort() {
    return (PortRegister*)getReg();
  }
};

class stimulus_symbol : public symbol
{
protected:
  stimulus *s;
public:
  stimulus_symbol(stimulus *);
  virtual string &name(void) const;
  virtual char *name(char *, int len);
  virtual string toString();
  virtual void new_name(const char *);
  virtual void new_name(string &);
  stimulus * getStimulus() {
    return s;
  }

};

class address_symbol : public Integer
{
public:

  address_symbol(const char *, unsigned int);
  virtual string toString();
  virtual Value* evaluate();
  /// breakpoints
  virtual int set_break(ObjectBreakTypes bt=eBreakAny, ObjectActionTypes at=eActionHalt, Expression *expr=0);
};

class line_number_symbol : public address_symbol
{
protected:
  int src_id,src_line,lst_id,lst_line,lst_page;
 public:

  line_number_symbol(const char *, unsigned int);
  void put_address(int new_address) {set(new_address);}
  void put_src_line(int new_src_line) {src_line = new_src_line;}
  void put_lst_line(int new_lst_line) {lst_line = new_lst_line;}
  void put_lst_page(int new_lst_page) {lst_page = new_lst_page;}
};

/// module_symbol - a symbol table entry for a gpsim module.
class module_symbol : public symbol
{
protected:
  Module *module;
public:
  module_symbol(Module *, const char *);
  virtual string toString();
  Module *get_module() { return module;}
  virtual Value *copy();
  virtual string description();

  /// The set and get methods for the module_symbol take char strings
  /// as their inputs. These strings are dynamically decoded such
  /// that various internal states of the processor can be controlled.

  virtual void set(const char *cP,int len=0);
  virtual void get(char *, int len);

};

/// attribute_symbol - a symbol that is associated with a specific
/// instance of a module. This class provides a mechanism for wrapping
/// Values (or symbols) and associating them with processors or
/// modules.

class attribute_symbol : public module_symbol
{
protected:
  /// The attribute wrapped by this symbol.
  Value *attribute;
public:
  attribute_symbol(Module *, Value *);
  virtual string toString();
  virtual string description();

  /// The get and set methods will call the attribute's get and
  /// set methods.

  virtual void set(double);
  virtual void set(gint64);
  virtual void set(int);
  virtual void set(Value *);
  virtual void set(Expression *);
  virtual void set(const char *cP,int len=0);
  virtual void set(Packet &);

  virtual void get(int &);
  virtual void get(gint64 &);
  virtual void get(double &);
  virtual void get(char *, int len);
  virtual void get(Packet &);
  virtual void get(Value **);

  /// copy the object value to a user char array
  virtual char *toString(char *, int len);
  virtual char *toBitStr(char *, int len);

  virtual void set_xref(Value *);
  virtual Value *get_xref();
  virtual Value *copy();
  virtual Value* evaluate();

};

// Place W into the symbol table
class w_symbol : public register_symbol
{
 public:
  w_symbol(const char*, Register *);
  virtual string toString();
};


class val_symbol : public symbol
{
 public:

  val_symbol(gpsimValue *);

  gpsimValue *val;

  virtual string toString();

  virtual void get(int &);
  virtual void get(gint64 &);
  virtual void set(int);
  virtual void set(gint64);
  virtual symbol *copy();
  virtual bool compare(ComparisonOperator *compOp, Value *rvalue);

  virtual string &name(void) const;

};

#endif  //  __SYMBOL_H__