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

📄 sc_string.cpp

📁 system C源码 一种替代verilog的语言
💻 CPP
字号:
/*****************************************************************************  The following code is derived, directly or indirectly, from the SystemC  source code Copyright (c) 1996-2006 by all Contributors.  All Rights reserved.  The contents of this file are subject to the restrictions and limitations  set forth in the SystemC Open Source License Version 2.4 (the "License");  You may not use this file except in compliance with such restrictions and  limitations. You may obtain instructions on how to receive a copy of the  License at http://www.systemc.org/. Software distributed by Contributors  under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF  ANY KIND, either express or implied. See the License for the specific  language governing rights and limitations under the License. *****************************************************************************//*****************************************************************************  sc_string.cpp -- Implementation of a simple string class.  Original Author: Stan Y. Liao, Synopsys, Inc. *****************************************************************************//*****************************************************************************  MODIFICATION LOG - modifiers, enter your name, affiliation, date and  changes you are making here.      Name, Affiliation, Date:  Description of Modification: *****************************************************************************/// $Log: sc_string.cpp,v $// Revision 1.1.1.1  2006/12/15 20:31:39  acg// SystemC 2.2//// Revision 1.3  2006/01/13 18:53:11  acg// Andy Goodrich: Added $Log command so that CVS comments are reproduced in// the source.//#include <assert.h>#include <ctype.h>#include <stdio.h>#include <stdarg.h>#include <string.h>#include "sysc/utils/sc_iostream.h"#include "sysc/utils/sc_string.h"#include "sysc/utils/sc_utils_ids.h"namespace sc_dt {inline intsc_roundup( int n, int m ){    return ((n - 1) / m + 1) * m;}// ----------------------------------------------------------------------------//  ENUM : sc_numrep////  Enumeration of number representations for character string conversion.// ----------------------------------------------------------------------------const std::stringto_string( sc_numrep numrep ){    switch( numrep )    {        case SC_DEC:	    return std::string( "SC_DEC" );        case SC_BIN:	    return std::string( "SC_BIN" );        case SC_BIN_US:	    return std::string( "SC_BIN_US" );        case SC_BIN_SM:	    return std::string( "SC_BIN_SM" );        case SC_OCT:	    return std::string( "SC_OCT" );        case SC_OCT_US:	    return std::string( "SC_OCT_US" );        case SC_OCT_SM:	    return std::string( "SC_OCT_SM" );        case SC_HEX:	    return std::string( "SC_HEX" );        case SC_HEX_US:	    return std::string( "SC_HEX_US" );        case SC_HEX_SM:	    return std::string( "SC_HEX_SM" );        case SC_CSD:	    return std::string( "SC_CSD" );	default:	    return std::string( "unknown" );    }}// ----------------------------------------------------------------------------//  CLASS : sc_string_rep////  Reference counting string implementation class.// ----------------------------------------------------------------------------class sc_string_rep{    friend class sc_string_old;    friend ::std::ostream& operator<<( ::std::ostream&, const sc_string_old& );    friend ::std::istream& operator>>( ::std::istream&, sc_string_old& );    friend sc_string_old operator+( const char*, const sc_string_old& );    sc_string_rep( int size = 16 )    {        ref_count = 1;        alloc = sc_roundup( size, 16 );        str = new char[alloc];        *str = '\0';    }    sc_string_rep( const char* s )    {        ref_count = 1;        if (s) {            alloc = 1 + strlen(s);            str = strcpy( new char[alloc], s );        }        else {            alloc = 16;            str = strcpy( new char[alloc], "" );        }    }    sc_string_rep( const char* s, int n); // get first n chars from the string    ~sc_string_rep()    {        assert( ref_count == 0 );        delete[] str;    }    void resize( int new_size );    void set_string( const char* s );    int ref_count;    int alloc;    char* str;};// IIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIIsc_string_rep::sc_string_rep( const char* s, int n){    ref_count = 1;    if (s && n>0) {        alloc = 1 + n;        str = strncpy( new char[alloc], s,n );        str[n] = 00;    }    else {        alloc = 16;        str = strcpy( new char[alloc], "" );    }}voidsc_string_rep::resize( int new_size ){    if (new_size <= alloc) return;    alloc = sc_roundup( new_size, 16 );    char* new_str = strcpy( new char[alloc], str );    delete[] str;    str = new_str;}voidsc_string_rep::set_string( const char* s ){    int len = strlen(s);    resize( len + 1 );    strcpy( str, s );}// ----------------------------------------------------------------------------//  CLASS : sc_string_old////  String class (yet another).// ----------------------------------------------------------------------------// constructorssc_string_old::sc_string_old( int size ){    rep = new sc_string_rep( size );}sc_string_old::sc_string_old( const char* s ){    rep = new sc_string_rep( s );}sc_string_old::sc_string_old( const char* s, int n ){    rep = new sc_string_rep( s, n );}sc_string_old::sc_string_old( const sc_string_old& s ){    rep = s.rep;    rep->ref_count ++;}sc_string_old::sc_string_old( sc_string_rep* r ){    rep = r;}// destructorsc_string_old::~sc_string_old(){    if( -- (rep->ref_count) == 0 ) {        delete rep;    }}intsc_string_old::length() const{    return strlen(rep->str);}sc_string_oldsc_string_old::operator+( const char* s ) const{    int len = length();    sc_string_rep* r = new sc_string_rep( len + strlen(s) + 1 );    strcpy( r->str, rep->str );    strcpy( r->str + len, s );    return sc_string_old(r);}sc_string_old sc_string_old::operator+(char c) const{    int len = length();    sc_string_rep* r = new sc_string_rep( len + 2 );    strcpy( r->str, rep->str );    r->str[len] = c;    r->str[len+1] = 00;    return sc_string_old(r);}sc_string_oldoperator+( const char* s, const sc_string_old& t ){    int len = strlen(s);    sc_string_rep* r = new sc_string_rep( len + t.length() + 1 );    strcpy( r->str, s );    strcpy( r->str + len, t );    return sc_string_old(r);}sc_string_oldsc_string_old::operator+( const sc_string_old& s ) const{    int len = length();    sc_string_rep* r = new sc_string_rep( len + s.length() + 1 );    strcpy( r->str, rep->str );    strcpy( r->str + len, s.rep->str );    return sc_string_old(r);}sc_string_old&sc_string_old::operator=( const char* s ){    if (rep->ref_count > 1) {        --rep->ref_count;        rep = new sc_string_rep(s);    }    else {        rep->set_string(s);    }    return *this;}sc_string_old&sc_string_old::operator=( const sc_string_old& s ){    if (&s == this)        return *this;    if (--(rep->ref_count) == 0)        delete rep;    rep = s.rep;    rep->ref_count++;    return *this;}sc_string_old&sc_string_old::operator+=( const char* s ){    int oldlen = length();    int slen   = strlen(s);    if (rep->ref_count > 1) {        sc_string_rep* oldrep = rep;        --rep->ref_count;        rep = new sc_string_rep( oldlen + slen + 1 );        strcpy( rep->str, oldrep->str );        strcpy( rep->str + oldlen, s );    }    else {        rep->resize( oldlen + slen + 1 );        strcpy( rep->str + oldlen, s );    }    return *this;}sc_string_old& sc_string_old::operator+=(char c){    int oldlen = length();    if (rep->ref_count > 1) {        sc_string_rep* oldrep = rep;        --rep->ref_count;        rep = new sc_string_rep( oldlen + 2 );        strcpy( rep->str, oldrep->str );        rep->str[oldlen]=c;        rep->str[oldlen+1]=00;    }    else {        rep->resize( oldlen + 2 );        rep->str[oldlen]=c;        rep->str[oldlen+1]=00;    }    return *this;}sc_string_old&sc_string_old::operator+=( const sc_string_old& s ){    return this->operator+=( s.rep->str );}intsc_string_old::cmp( const char* s ) const{    return strcmp( rep->str, s );}intsc_string_old::cmp( const sc_string_old& s ) const{    return strcmp( rep->str, s.rep->str );}const char* sc_string_old::c_str() const{  return rep->str;}// get substringsc_string_old sc_string_old::substr(int first,int last) const{  if(first<0 || last<0 || first>last || first>=length() || last>=length())    return "";  return sc_string_old(rep->str+first, last-first+1);}sc_string_old sc_string_old::make_str(long n) // convert integer to string{  char buf[32];  ::std::sprintf(buf,"%ld",n);  return sc_string_old(buf);}#define DEFINE_RELOP(op) \bool sc_string_old::operator op( const char* s ) const \{						\    return strcmp( rep->str, s ) op 0;		\}						\bool sc_string_old::operator op( const sc_string_old& s ) const \{						\    return strcmp( rep->str, s.rep->str ) op 0;	\}DEFINE_RELOP(==)DEFINE_RELOP(!=)DEFINE_RELOP(<)DEFINE_RELOP(<=)DEFINE_RELOP(>)DEFINE_RELOP(>=)sc_string_old::operator const char*() const{    return rep->str;}charsc_string_old::operator[]( int i ) const{    return rep->str[i];}char& sc_string_old::operator[]( int i ){    if (rep->ref_count > 1) {        rep->ref_count--;        rep = new sc_string_rep(rep->str);    }    return rep->str[i];}voidsc_string_old::set( int i, char c ){    if (rep->ref_count > 1) {        rep->ref_count--;        rep = new sc_string_rep(rep->str);    }    rep->str[i] = c;}sc_string_old sc_string_old::to_string(const char* format, ...){   va_list argptr;   int cnt;   sc_string_old result;   char buffer[1024]; // static string buffer   buffer[1023]=000;   va_start(argptr, format);#if defined(WIN32)   // Windows provides safer implementation#if defined(_MSC_VER)   cnt = _vsnprintf(buffer, 1024, format, argptr);#else   cnt = vsnprintf(buffer, 1024, format, argptr);#endif   if(cnt>1023) // string too long   {     int buf_size = 1024;     const int max_size = 65000;     char* buf; // dynamic buffer     do     {       buf_size*=2;       buf = new char[buf_size];#if defined(_MSC_VER)       cnt = _vsnprintf(buffer, buf_size, format, argptr);#else       cnt = vsnprintf(buffer, buf_size, format, argptr);#endif       if(buf_size<max_size && cnt>=buf_size)         delete[] buf;     }     while( buf_size<max_size && cnt>=buf_size);     if(cnt>=buf_size)     {       // string is longer the the maximum buffer size (max_size)       SC_REPORT_WARNING( sc_core::SC_ID_STRING_TOO_LONG_, "truncated" );       buf[buf_size-1] = 000;     }     result = buf;     delete[] buf;   }   else     result = buffer;#else   try {     // this may end up in a core dump     // if we are lucky we can catch exception     cnt = vsprintf(buffer, format, argptr);   }   catch(...)   {     SC_REPORT_WARNING( sc_core::SC_ID_STRING_TOO_LONG_,			"program may become unstable" );   }   buffer[1023]=000; // in case it's longer   result = buffer;#endif   va_end(argptr);   return result;}voidsc_string_old::print( ::std::ostream& os ) const{    os << rep->str;}void sc_string_old::test(int position)const{	if(position<0 || position>=length())		SC_REPORT_ERROR( sc_core::SC_ID_OUT_OF_BOUNDS_, "sc_string_old::test" );}// TODO: conveniece formatting functions for common types//       e.g. sc_string_old("a=%d, s is %s").fmt(1).fmt("string")//       should produce a=1, s is string//       it should be safe: if less arguments specified//       it should print %specifier; extra arguments should be ignored//       if the type of the argument is incompatible with format //       specifier it should be ignored//unsignedsc_string_old::fmt_length()const{    unsigned result=0;    if((*this)[0]!='%')	return 0;    else	result++;    if(is_delimiter("-+0 #",result)) // flags	result++;    while(is_delimiter("0123456789*",result)) // width	result++;    if(rep->str[result]=='.') // precision    {	result++;	unsigned old_result = result;	while(is_delimiter("0123456789*",result)) result++;	if(old_result == result) //error in format	    return 0;    }    if(is_delimiter("hlL",result)) result++; // I64 is not supported    if(is_delimiter("cCdiouxXeEfgGnpsS",result)) 	result++;    else // error in format	return 0;    return result;}sc_string_old&sc_string_old::fmt(const sc_string_old& s){    return fmt(s.c_str());}intsc_string_old::pos( const sc_string_old& sub_string ) const{    int sub_len = sub_string.length();    if( sub_len == 0 ) {        return 0; // empty string always matches    }    int ind = 0;    int len = length();    bool found = false;    while( ind < len && ! found )    {        found = ( sub_string == substr( ind, ind + sub_len - 1 ) );        ++ ind;    }    if( found ) {        return -- ind;    } else {        return -1;    }}sc_string_old&sc_string_old::remove(unsigned index, unsigned length){    test((int)index);    if(length!=0)	(*this) = substr(0,index-1) + substr(index+length,this->length()-1);    return *this;}sc_string_old&sc_string_old::insert(const sc_string_old& sub_string, unsigned index){    if(index>(unsigned)length())   	SC_REPORT_ERROR( sc_core::SC_ID_OUT_OF_BOUNDS_, "sc_string_old::insert" );    return (*this) = substr(0,index-1)+sub_string+substr(index,length()-1);}boolsc_string_old::is_delimiter(const sc_string_old& str, unsigned index)const{    test((int)index);    return str.contains(rep->str[index]);}boolsc_string_old::contains(char c)const{    int len = length();    int i=0;    bool found = false;    while(!found && i<len)	found = rep->str[i++]==c;    return found;}sc_string_oldsc_string_old::uppercase()const{    int len = length();    sc_string_old temp(*this);    for(int i=0; i<len; i++)    {	char c = temp.rep->str[i];	if(c>='a' && c<='z')	    temp.rep->str[i] = static_cast<char>( c-32 );    }    return temp;}sc_string_oldsc_string_old::lowercase()const{    int len = length();    sc_string_old temp(*this);    for(int i=0; i<len; i++)    {	char c = temp.rep->str[i];	if(c>='A' && c<='Z')	    temp.rep->str[i] = static_cast<char>( c+32 );    }    return temp;}// ----------------------------------------------------------------------------::std::istream&operator >> ( ::std::istream& is, sc_string_old& s ){    if( s.rep->ref_count > 1 ) {        -- s.rep->ref_count;        s.rep = new sc_string_rep;    }    int i = 0;    char* p = s.rep->str;    char c;    // skip white spaces    while( is.get( c ) && isspace( c ) )        ;    for( ; is.good() && ! isspace( c ); is.get( c ) ) {        if( i > s.rep->alloc - 2 ) {	    s.rep->str[i] = '\0';            s.rep->resize( (int) (s.rep->alloc * 1.5) );            p = s.rep->str + i;        }        *p ++ = c;        i ++;    }    *p = '\0';    return is;} } // namespace sc_dt

⌨️ 快捷键说明

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