⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 archive.cpp

📁 c-smile 一个语法类似与JS 又有点像C++的 编译器
💻 CPP
📖 第 1 页 / 共 2 页
字号:
/*
*
* archive.cpp
*
* Copyright (c) 2001, 2002
* Andrew Fedoniouk - andrew@terra-informatica.org
* Portions: Serge Kuznetsov -  kuznetsov@deeptown.org
*
* See the file "COPYING" for information on usage 
* and redistribution of this file
*
*/
#include "c-smile.h"
#include "vm.h"
#include "streams.h"
#include "osfile.h"

namespace c_smile
{
  symbol_t
    archive::map ( symbol_t extern_sym )
  {
    if ( extern_sym == undefined_symbol )
      return undefined_symbol;
    for ( int i = 0; i < _symbols.size(); ++i )
      if ( _symbols [ i ] == extern_sym )
        return symbol_t ( i );

    _symbols.push ( extern_sym );
    return symbol_t ( _symbols.size () - 1 );
  }

  void
    archive::write ( const char *buf, size_t sz )
  {
    int s ( sz );
    if ( !_body->write ( buf, s ) || (size_t) s != sz )
      VM::error ( (char *) (const char *) _body->err_message );
  }

  void
    archive::read ( char *buf, size_t sz )
  {
    int s ( sz );
    if ( !_body->read ( buf, s ) || (size_t) s != sz )
      VM::error ( (char *) (const char *) _body->err_message );
  }

  void
    archive::write_class_ref ( const CLASS *cls )
  {
    symbol_t package_name = ( ( cls->package ) ?
                                cls->package->name : undefined_symbol );
    symbol_t class_name = cls->name;
    write ( package_name );
    write ( class_name );
  }

  void
    archive::read_class_ref ( CLASS * &cls )
  {
    symbol_t package_name, class_name;
    read ( package_name );
    read ( class_name );
    if ( package_name == undefined_symbol )
    {
      // package
      cls = VM::find_package ( class_name );
    }
    else
    {
      cls = VM::find_class ( package_name, class_name );
    }
  }

  //|
  //| bool
  //|
  void
    archive::read ( bool & v )
  {
    read ( (char *) &v, sizeof ( v ) );
  }

  void
    archive::write ( bool v )
  {
    write ( (char *) &v, sizeof ( v ) );
  }

  //|
  //| int
  //|
  void
    archive::read ( int & v )
  {
    read ( (char *) &v, sizeof ( v ) );
  }

  void
    archive::write ( int v )
  {
    write ( (char *) &v, sizeof ( v ) );
  }

  //|
  //| double
  //|
  void
    archive::read ( double & v )
  {
    read ( (char *) &v, sizeof ( v ) );
  }

  void
    archive::write ( double v )
  {
    write ( (char *) &v, sizeof ( v ) );
  }

  //|
  //| string
  //|
  void
    archive::read ( string & s )
  {
    int l;
    read ( l );
    assert ( l < 1024 );
    string ss ( ' ', l );
    read ( ss.buffer(), l );
    s = ss;
  }

  void
    archive::write ( const char *s )
  {
    int l = strlen ( s );
    write ( l );
    write ( s, l );
  }

  //|
  //| DICTIONARY
  //|
  void
    archive::read ( DICTIONARY* &dict )
  {
    int sz = 0;
    read ( sz );
    if ( sz == -1 )
      dict = 0;
    else
    {
      dict = DICTIONARY::create ( sz );
      for ( int i = 0; i < sz; i++ )
      {
        DICTIONARY::item& it = ( *dict ) [ i ];
        read ( it.symbol );
        read ( it.type   );
        read ( it.value  );
      }
    }
  }

  void
    archive::write ( const DICTIONARY * dict )
  {
    if ( dict == 0 )
      write ( -1 );
    else
    {
      write ( dict->size () );
      for ( int i = 0; i < dict->size(); i++ )
      {
        const DICTIONARY::item& it = ( *dict ) [ i ];
        write ( it.symbol );
        write ( it.type   );
        write ( it.value  );
      }
    }
  }

  //|
  //| STRING
  //|
  void
    archive::write ( const STRING * str )
  {
    if ( str == 0 )
      write ( -1 );
    else
    {
      int sz = (int) str->size();
      write ( sz );
      if ( sz )
        write ( CSTR ( str ), (int) str->size() );
    }
  }

  void
    archive::read ( STRING* &str )
  {
    int l = 0;
    read ( l );
    if ( l == -1 )
      str = 0;
    else
    {
      string s ( ' ', l );
      read ( s.buffer(), l );
      str = new STRING ( s );
    }
  }

  //|
  //| ARRAY
  //|
  void
    archive::write ( const ARRAY * vec )
  {
    if ( vec == 0 )
      write ( -1 );
    else
    {
      write ( vec->size() );
      for ( int i = 0; i < vec->size(); ++i )
        write ( ( *vec ) [ i ] );
    }
  }

  void
    archive::read ( ARRAY* &vec )
  {
    int l;
    read ( l );
    if ( l < 0 )
      vec = 0;
    else
    {
      vec = new ARRAY ( l );
      for ( int i = 0; i < l; ++i )
        read ( ( *vec ) [ i ] );
    }
  }

  //|
  //| CLASS
  //|
#define PACKAGE_EXISTS 0x01
#define BASE_EXISTS 0x02

  void
    archive::read_class ( CLASS *cls )
  {
    int i_sz = 0;
    read ( i_sz );

    cls->instance_size = i_sz;

    int flags = 0;
    read ( flags );
    if ( flags & PACKAGE_EXISTS )
    {
      symbol_t sym = undefined_symbol;
      read ( sym );
      cls->package = VM::find_package ( sym );
    }
    if ( flags & BASE_EXISTS )
    {
      read_class_ref ( cls->base );
    }
    read ( cls->members );
    for ( int i = 0; i < cls->members->size(); i++ )
    {
      DICTIONARY::item& it = ( *cls->members ) [ i ];
      if ( it.value.is_bytecode() )
      {
        CODE *pbc = it.value.v.v_code;
        pbc->_klass = cls;
        pbc->_name = it.symbol;

        cls->check_name ( pbc->_name, pbc, it.type );
      }
    }
  }

  void
    archive::write_class ( const CLASS *cls )
  {
    write ( (int) cls->instance_size );

    int flags = 0;
    if ( cls->package )
      flags |= PACKAGE_EXISTS;
    if ( cls->base )
      flags |= BASE_EXISTS;

    write ( flags );

    if ( cls->package )
      write ( cls->package->name );
    if ( cls->base )
      write_class_ref ( cls->base );

    write ( cls->members );
  }

  void
    archive::write ( const CLASS * cls )
  {
    write ( cls->name );
    write_class ( cls );
  }

  void
    archive::read ( CLASS* &cls )
  {
    symbol_t t;
    read ( t );
    cls = new CLASS ();
    cls->name = t;
    read_class ( cls );
  }

  //|
  //| PACKAGE
  //|
  void
    archive::write ( const PACKAGE * pkg )
  {
    write ( pkg->name );
    write_class ( (CLASS *) pkg );

    write ( pkg->literals->size () );
    for ( int i = 0; i < pkg->literals->size (); i++ )
      write_literal ( ( *pkg->literals ) [ i ] );

    write ( pkg->init_code );
    write ( pkg->file_name );
  }

  void
    archive::read ( PACKAGE* &pkg )
  {
    symbol_t t;
    read ( t );
    pkg = new PACKAGE ( t );
    VM::add_package ( pkg );
    
    read_class ( (CLASS *) pkg );

    int sz = 0;
    read ( sz );
    pkg->literals = new ARRAY ( sz );
    for ( int i = 0; i < sz; i++ )
    read_literal ( ( *pkg->literals ) [ i ] );

    read ( pkg->init_code );
    pkg->init_code->klass ( pkg );
    read ( pkg->file_name );

    VM::run_init_code ( pkg );

  }

  //|
  //| BYTECODE
  //|
  void
    archive::read ( CODE* &c )
  {
    c = new CODE ();
    bool bc = false;
    read ( bc );
    c->_is_native = !bc;
    if ( bc )
    {
      int sz = 0; read ( sz );
      c->_code.bc = new BUFFER ( sz );
      read ( (char *) &( *c->_code.bc ) [ 0 ], sz );
    }
  }

  //|
  //| BYTECODE
  //|
  void
    archive::read_code_ref ( CODE* &c )
  {
    CLASS *cls = 0;
    read_class_ref ( cls );
    assert ( cls );
    symbol_t sym = undefined_symbol;
    read ( sym );
    ENTRY e = cls->find ( sym );
    assert ( e.is_valid () );
    assert ( e.type () == ST_FUNCTION || e.type () == ST_SFUNCTION );
    c = e.value ()->v.v_code;
  }

  void
    archive::write ( const CODE* c )
  {
    write ( c->is_bytecode () );
    if ( c->is_bytecode () )
    {
      write ( (int) c->_code.bc->size () );
      write ( (char *) &( *c->_code.bc ) [ 0 ], c->_code.bc->size () );
    }
  }

  void
    archive::write_code_ref ( const CODE* c )
  {
    write_class_ref ( c->_klass );
    write ( c->_name );
  }

  //|
  //| ENTRY
  //|
  void
    archive::read ( ENTRY &var )
  {
    read_class_ref ( var.klass );
    int idx = 0;
    read ( var.index );
  }

  void
    archive::write ( const ENTRY &var )
  {
    write_class_ref ( var.klass );
    write ( var.index );
  }

  //|
  //| SYMBOL
  //|
  void
    archive::read ( symbol_t& sym )
  {
    int i;
    read ( i ); //i - local (archive index);
    if ( (symbol_t) i == undefined_symbol )
      sym = undefined_symbol;
    else sym = _symbols [ i ];
  }
  
  void
    archive::write ( symbol_t sym )
  {
    int i = map ( sym );
    write ( i );
  }

  //|
  //| OBJECT
  //|
  void
    archive::read ( OBJECT* &obj )
  {
    CLASS *cls;
    read_class_ref ( cls );
    obj = new OBJECT ( cls );
    for ( int i = 0; i < obj->klass->instance_size; i++ )
      read ( obj->members [ i ] );
  }

  void
    archive::write ( const OBJECT * obj )
  {
    write_class_ref ( obj->klass );
    for ( int i = 0; i < obj->klass->instance_size; i++ )
      write ( obj->members [ i ] );
  }

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -