Diff for /parser3/src/main/pa_string.C between versions 1.8 and 1.113

version 1.8, 2001/01/27 12:04:53 version 1.113, 2001/10/24 09:03:42
Line 1 Line 1
 /*  /** @file
   $Id$          Parser: string class. @see untasize_t.C.
   
           Copyright (c) 2001 ArtLebedev Group (http://www.artlebedev.com)
           Author: Alexander Petrosyan <paf@design.ru> (http://design.ru/paf)
   
           $Id$
 */  */
   
 #include <string.h>  #include "pa_config_includes.h"
   
   #include "pcre.h"
   #include "internal.h"
   
 #include "pa_pool.h"  #include "pa_pool.h"
   #include "pa_string.h"
 #include "pa_hash.h"  #include "pa_hash.h"
   #include "pa_exception.h"
 void *String::operator new(size_t size, Pool *apool) {  #include "pa_common.h"
         return apool->malloc(size);  #include "pa_array.h"
 }  #include "pa_globals.h"
   #include "pa_table.h"
 void String::construct(Pool *apool) {  #include "pa_dictionary.h"
         pool=apool;  
         head.count=curr_chunk_rows=CR_PREALLOCATED_COUNT;  String::String(Pool& apool, const char *src, size_t src_size, bool tainted) :
           Pooled(apool),
           forigins_mode(false) {
           last_chunk=&head;
           head.count=CR_PREALLOCATED_COUNT;
         append_here=head.rows;          append_here=head.rows;
         head.preallocated_link=0;          head.preallocated_link=0;
         link_row=&head.rows[curr_chunk_rows];          link_row=&head.rows[head.count];
         fused_rows=fsize=0;          fused_rows=fsize=0;
 }  
   
 void String::expand() {          if(src)
         curr_chunk_rows+=curr_chunk_rows*CR_GROW_PERCENT/100;                  if(tainted)
         Chunk *chunk=static_cast<Chunk *>(                          APPEND_TAINTED(src, src_size, 0, 0);
                 pool->malloc(sizeof(Chunk::Row)*curr_chunk_rows+sizeof(Chunk *)));                  else
         chunk->count=curr_chunk_rows;                          APPEND_CLEAN(src, src_size, 0, 0);
         link_row->link=chunk;  
         append_here=chunk->rows;  
         link_row=&chunk->rows[curr_chunk_rows];  
         link_row->link=0;  
 }  }
   
 String::String(String& src) {  String::String(const String& src) :     
         pool=src.pool;          Pooled(src.pool()),
           forigins_mode(false) {
         head.count=CR_PREALLOCATED_COUNT;          head.count=CR_PREALLOCATED_COUNT;
                   
         int src_used_rows=src.used_rows();          size_t src_used_rows=src.fused_rows;
         if(src_used_rows<=head.count) {          if(src_used_rows<=head.count) {
                 // new rows fit into preallocated area                  // all new rows fit size_to preallocated area
                 curr_chunk_rows=head.count;                  last_chunk=&head;
                   size_t curr_chunk_rows=head.count;
                 memcpy(head.rows, src.head.rows, sizeof(Chunk::Row)*src_used_rows);                  memcpy(head.rows, src.head.rows, sizeof(Chunk::Row)*src_used_rows);
                 append_here=&head.rows[src_used_rows];                  append_here=&head.rows[src_used_rows];
                 head.preallocated_link=0;  
                 link_row=&head.rows[curr_chunk_rows];                  link_row=&head.rows[curr_chunk_rows];
         } else {          } else {
                 // warning:                   // warning: 
                 //   heavy relies on the fact                   //   heavily relies on the fact 
                 //   that preallocated area is the same for all strings                  //   "preallocated area is the same for all strings"
                 //                  //
                 // info:                  // info:
                 //   allocating only enough mem to fit src string rows                  //   allocating only enough mem to fit src string rows
                 //   next append would allocate a new chunk                  //   next append would allocate a new chunk
                 //                  //
                 // new rows don't fit into preallocated area: splitting into two chunks                  // new rows don't fit size_to preallocated area: splitting size_to two chunks
                 // preallocated chunk src to constructing head                  // preallocated chunk src to constructing head
                 memcpy(head.rows, src.head.rows, sizeof(Chunk::Row)*head.count);                  memcpy(head.rows, src.head.rows, sizeof(Chunk::Row)*head.count);
                 // remaining rows into new_chunk                  // remaining rows size_to new_chunk
                 curr_chunk_rows=src_used_rows-head.count;                  size_t curr_chunk_rows=src_used_rows-head.count;
                 Chunk *new_chunk=static_cast<Chunk *>(                  last_chunk=static_cast<Chunk *>(
                         pool->malloc(sizeof(Chunk::Row)*curr_chunk_rows+sizeof(Chunk *)));                          malloc(sizeof(size_t)+sizeof(Chunk::Row)*curr_chunk_rows+sizeof(Chunk *)));
                 new_chunk->count=curr_chunk_rows;                  last_chunk->count=curr_chunk_rows;
                 head.preallocated_link=new_chunk;                  head.preallocated_link=last_chunk;
                 append_here=link_row=&new_chunk->rows[curr_chunk_rows];                  append_here=link_row=&last_chunk->rows[last_chunk->count];
   
                 Chunk *old_chunk=src.head.preallocated_link;                   Chunk *old_chunk=src.head.preallocated_link; 
                 Chunk::Row *new_rows=new_chunk->rows;                  Chunk::Row *new_rows=last_chunk->rows;
                 int rows_left_to_copy=curr_chunk_rows;                  size_t rows_left_to_copy=last_chunk->count;
                 while(true) {                  while(true) {
                         int old_count=old_chunk->count;                          size_t old_count=old_chunk->count;
                         Chunk *next_chunk=old_chunk->rows[old_count].link;                          Chunk *next_chunk=old_chunk->rows[old_count].link;
                         if(next_chunk) {                          if(next_chunk) {
                                 // not last source chunk                                  // not last source chunk
Line 90  String::String(String& src) { Line 99  String::String(String& src) {
         fsize=src.fsize;          fsize=src.fsize;
 }  }
   
 String& String::operator += (char *src) {  void String::expand() {
           size_t new_chunk_count=last_chunk->count+CR_GROW_COUNT;
           last_chunk=static_cast<Chunk *>(
                   malloc(sizeof(size_t)+sizeof(Chunk::Row)*new_chunk_count+sizeof(Chunk *)));
           last_chunk->count=new_chunk_count;
           link_row->link=last_chunk;
           append_here=last_chunk->rows;
           link_row=&last_chunk->rows[last_chunk->count];
           link_row->link=0;
   }
   
   String& String::append(const String& src, Untaint_lang lang, bool forced) {
           const Chunk *chunk=&src.head; 
           do {
                   const Chunk::Row *row=chunk->rows;
                   for(size_t i=0; i<chunk->count; i++, row++) {
                           if(row==src.append_here)
                                   goto break2;
                           
                           APPEND(row->item.ptr, row->item.size, 
                                   (lang!=UL_PASS_APPENDED && (row->item.lang==UL_TAINTED || forced))?lang:row->item.lang,
                                   row->item.origin.file, row->item.origin.line);
                   }
                   chunk=row->link;
           } while(chunk);
   break2:
           return *this;
   }
   
   String& String::real_append(STRING_APPEND_PARAMS) {
           if(!src)
                   return *this;
           if(!size)
                   size=strlen(src);
           if(!size)
                   return *this;
   
         if(chunk_is_full())          if(chunk_is_full())
                 expand();                  expand();
   
         append_here->item.ptr=src;          append_here->item.ptr=src;
         fsize+=append_here->item.size=strlen(src);          fsize+=append_here->item.size=size;
           append_here->item.lang=lang;
   #ifndef NO_STRING_ORIGIN
           append_here->item.origin.file=file;
           append_here->item.origin.line=line;
   #endif
         append_here++; fused_rows++;          append_here++; fused_rows++;
   
         return *this;          return *this;
 }  }
   
 char *String::c_str() {  char String::first_char() const {
         char *result=static_cast<char *>(pool->malloc(size()+1));          if(!fused_rows)
                   throw Exception(0, 0,
                           this,
                           "getting first char of empty string");
   
         char *copy_here=result;          return *head.rows[0].item.ptr;
         Chunk *chunk=&head;   }
   
   uint String::hash_code() const {
           uint result=0;
   
           const Chunk *chunk=&head; 
         do {          do {
                 Chunk::Row *row=chunk->rows;                  const Chunk::Row *row=chunk->rows;
                 for(int i=0; i<chunk->count; i++) {                  for(size_t i=0; i<chunk->count; i++) {
                         if(row==append_here)                          if(row==append_here)
                                 goto break2;                                  goto break2;
   
                         memcpy(copy_here, row->item.ptr, row->item.size);                          result=Hash::generic_code(result, row->item.ptr, row->item.size);
                         copy_here+=row->item.size;  
                         row++;                          row++;
                 }                  }
                 chunk=row->link;                  chunk=row->link;
         } while(chunk);          } while(chunk);
 break2:  break2:
         *copy_here=0;  
         return result;          return result;
 }  }
   
 uint String::hash_code() {  /// @todo move 'lang' skipping to pos
         uint result=0;  int String::cmp(int& partial, const String& src, 
                                   size_t this_offset, Untaint_lang lang) const {
           partial=-1;
           this_offset=min(this_offset, size()-1);
   
           const Chunk *a_chunk=&head;
           const Chunk *b_chunk=&src.head;
           const Chunk::Row *a_row=a_chunk->rows;
           const Chunk::Row *b_row=b_chunk->rows;
           size_t a_offset=this_offset;
           size_t b_offset=0;
           Chunk::Row *a_end=append_here;
           Chunk::Row *b_end=src.append_here;
           size_t a_countdown=a_chunk->count;
           size_t b_countdown=b_chunk->count;
           size_t result;
           size_t pos=0; 
   
           bool a_break=size()==0;
           bool b_break=src.size()==0;
           if(!(a_break || b_break)) while(true) {
                   if(pos+a_row->item.size > this_offset) {
                           if(lang!=UL_UNSPECIFIED && a_row->item.lang!=lang) 
                                   return -1; // wrong lang -- bail out
   
                           int size_diff=
                                   (a_row->item.size-a_offset)-
                                   (b_row->item.size-b_offset);
                           
                           if(size_diff==0) { // a has same size as b
                                   result=memcmp(a_row->item.ptr+a_offset, b_row->item.ptr+b_offset, 
                                           a_row->item.size-a_offset);
                                   if(result)
                                           return result;
                                   pos+=a_row->item.size;
                                   a_row++; a_countdown--; a_offset=0;
                                   b_row++; b_countdown--; b_offset=0;
                           } else if (size_diff>0) { // a longer
                                   result=memcmp(a_row->item.ptr+a_offset, b_row->item.ptr+b_offset, 
                                           b_row->item.size-b_offset);
                                   if(result)
                                           return result;
                                   a_offset+=b_row->item.size-b_offset;
                                   b_row++; b_countdown--; b_offset=0;
                           } else { // b longer
                                   result=memcmp(a_row->item.ptr+a_offset, b_row->item.ptr+b_offset, 
                                           a_row->item.size-a_offset);
                                   if(result)
                                           return result;
                                   b_offset+=a_row->item.size-a_offset;
                                   pos+=a_row->item.size;
                                   a_row++; a_countdown--; a_offset=0;
                           }
                           if(b_break=b_row==b_end) {
                                   a_break=a_row==a_end;
                                   break;                  
                           }
                           if(!b_countdown) {
                                   b_chunk=b_row->link;
                                   b_row=b_chunk->rows;
                                   b_countdown=b_chunk->count;
                           }
                   } else {
                           a_offset-=a_row->item.size;
                           pos+=a_row->item.size;
                           a_row++; a_countdown--; 
                   }
   
                   if(a_break=a_row==a_end) {
                           b_break=b_row==b_end;
                           break;
                   }
                   if(!a_countdown) {
                           a_chunk=a_row->link;
                           a_row=a_chunk->rows;
                           a_countdown=a_chunk->count;
                   }
           }
           if(a_break==b_break) { // ended simultaneously
                   partial=0; return 0;
           } else if(a_break) { // first bytes equal, but a ended before b
                   partial=1; return -1;
           } else {
                   partial=2; return +1;
           }
   }
   
   /// @todo move 'lang' skipping to pos
   int String::cmp(int& partial, const char* b_ptr, size_t src_size, 
                                   size_t this_offset, Untaint_lang lang) const {
           partial=-1;
           size_t b_size=src_size?src_size:b_ptr?strlen(b_ptr):0;
           this_offset=min(this_offset, size()-1);
   
           const Chunk *a_chunk=&head;
           const Chunk::Row *a_row=a_chunk->rows;
           size_t a_offset=this_offset;
           size_t b_offset=0;
           Chunk::Row *a_end=append_here;
           size_t a_countdown=a_chunk->count;
           size_t pos=0;
   
           bool a_break=size()==0;
           bool b_break=b_size==0;
           if(!(a_break || b_break)) while(true) {
                   if(pos+a_row->item.size > this_offset) {
                           if(lang!=UL_UNSPECIFIED && a_row->item.lang!=lang) 
                                   return -1; // wrong lang -- bail out
   
                           int size_diff=
                                   (a_row->item.size-a_offset)-
                                   (b_size-b_offset);
                           
                           if(size_diff==0) { // a has same size as b
                                   if(size_t result=memcmp(a_row->item.ptr+a_offset, b_ptr+b_offset, 
                                           a_row->item.size-a_offset)!=0)
                                           return result;
                                   pos+=a_row->item.size;
                                   a_row++; a_countdown--; a_offset=0;
                                   b_break=true;
                           } else if (size_diff>0) { // a longer
                                   if(size_t result=memcmp(a_row->item.ptr+a_offset, b_ptr+b_offset, 
                                           b_size-b_offset)!=0)
                                           return result;
                                   a_offset+=b_size-b_offset;
                                   b_break=true;
                           } else { // b longer
                                   if(size_t result=memcmp(a_row->item.ptr+a_offset, b_ptr+b_offset, 
                                           a_row->item.size-a_offset)!=0)
                                           return result;
                                   b_offset+=a_row->item.size-a_offset;
                                   pos+=a_row->item.size;
                                   a_row++; a_countdown--; a_offset=0;
                           }
                   } else {
                           a_offset-=a_row->item.size; 
                           pos+=a_row->item.size;
                           a_row++; a_countdown--; 
                   }
   
                   a_break=a_row==a_end;
                   if(a_break || b_break)
                           break;
                   if(!a_countdown) {
                           a_chunk=a_row->link;
                           a_row=a_chunk->rows;
                           a_countdown=a_chunk->count;
                   }
           }
           if(a_break==b_break) { // ended simultaneously
                   partial=0; return 0;
           } else if(a_break) { // first bytes equal, but a ended before b
                   partial=1; return -1;
           } else {
                   partial=2; return +1;
           }
   }
   
   #ifndef NO_STRING_ORIGIN
   const Origin& String::origin() const { 
           if(!fused_rows) {
                   static const Origin empty_origin={"empty string"};
                   return empty_origin;
           }
           
           // determining origin by last appended piece
           // because first one frequently constant. 
           // ex: ^load[/file] "document_root" + "/file"
           // when last peice is constant, 
           // ex: parser_root_auto_path{dynamic} / auto.p{const}
           // using first piece
           Origin& last_origin=append_here[-1].item.origin;
           return last_origin.file ? last_origin : head.rows[0].item.origin;
   }
   #endif
   
         Chunk *chunk=&head;   String& String::mid(size_t start, size_t finish) const {
           String& result=*NEW String(pool());
   
           start=max(0, start);
           finish=min(size(), finish);
           if(start==finish)
                   return result;
   
           size_t pos=0;
           const Chunk *chunk=&head; 
         do {          do {
                 Chunk::Row *row=chunk->rows;                  const Chunk::Row *row=chunk->rows;
                 for(int i=0; i<chunk->count; i++) {                  for(size_t i=0; i<chunk->count; pos+=row->item.size, i++, row++) {
                         if(row==append_here)                          if(row==append_here)
                                 goto break2;                                  goto break2;
   
                         result=Hash::generic_code(result, row->item.ptr, row->item.size);                          size_t item_finish=pos+row->item.size;
                         row++;                          if(item_finish > start) { // started now or already?
                                   bool started=result.size()==0; // started now?
                                   bool finished=finish <= item_finish; // finished now?
                                   size_t offset=started?start-pos:0;
                                   size_t size=finished?finish-pos:row->item.size;
                                   result.APPEND(
                                           row->item.ptr+offset, size-offset, 
                                           row->item.lang,
                                           row->item.origin.file, row->item.origin.line);
                                   if(finished)
                                           goto break2;
                           }
                 }                  }
                 chunk=row->link;                  chunk=row->link;
         } while(chunk);          } while(chunk);
 break2:  break2:
   //      SAPI::log(pool(), "piece of '%s' from %d to %d is '%s'",
                   //cstr(), start, finish, result.cstr());
         return result;          return result;
 }  }
   
 bool String::operator == (String& src) {  int String::pos(const String& substr, 
         if(size() != src.size())                                  size_t result, Untaint_lang lang) const {
                 return false;          for(; result<size(); result++) {
                   int partial; cmp(partial, substr, result, lang);
         // FIX: 0 approach!                  if(
         // use: in Hash it's "this" that has less chunks                          partial==0 || // full match
         if(head.rows[0].item.size==src.head.rows[0].item.size)                          partial==2) // 'substr' starts 'this'+'result'
                 if(memcmp(head.rows[0].item.ptr, src.head.rows[0].item.ptr, head.rows[0].item.size)==0)                          return result;
           }
           
           return -1;
   }
   
   int String::pos(const char *substr, size_t substr_size, 
                                   size_t result, Untaint_lang lang) const {
           for(; result<size(); result++) {
                   int partial; cmp(partial, substr, substr_size, result, lang);
                   if(
                           partial==0 || // full match
                           partial==2) // 'substr' starts 'this'+'result'
                           return result;
           }
           
           return -1;
   }
   
   void String::split(Array& result, 
                                      size_t* pos_after_ref, 
                                      const char *delim, size_t delim_size, 
                                      Untaint_lang lang, int limit) const {
           if(delim_size) {
                   size_t pos_after=pos_after_ref?*pos_after_ref:0;
                   int pos_before;
                   // while we have 'delim'...
                   for(; (pos_before=pos(delim, delim_size, pos_after, lang))>=0 && limit; limit--) {
                           result+=&mid(pos_after, pos_before);
                           pos_after=pos_before+delim_size;
                   }
                   // last piece
                   if(pos_after<size() && limit) {
                           result+=&mid(pos_after, size());
                           pos_after=size();
                   }
                   if(pos_after_ref)
                           *pos_after_ref=pos_after;
           } else { // empty delim
                   result+=this;
                   if(pos_after_ref)
                           *pos_after_ref+=size();
           }
   }
   
   void String::split(Array& result, 
                                      size_t* pos_after_ref, 
                                      const String& delim, Untaint_lang lang, 
                                      int limit) const {
           if(delim.size()) {
                   size_t pos_after=pos_after_ref?*pos_after_ref:0;
                   int pos_before;
                   // while we have 'delim'...
                   for(; (pos_before=pos(delim, pos_after, lang))>=0 && limit; limit--) {
                           result+=&mid(pos_after, pos_before);
                           pos_after=pos_before+delim.size();
                   }
                   // last piece
                   if(pos_after<size() && limit) {
                           result+=&mid(pos_after, size());
                           pos_after=size();
                   }
                   if(pos_after_ref)
                           *pos_after_ref=pos_after;
           } else { // empty delim
                   result+=this;
                   if(pos_after_ref)
                           *pos_after_ref+=size();
           }
   }
   
   static void regex_options(char *options, int *result){
       struct Regex_option {
                   char key;
                   int clear, set;
                   int *result;
       } regex_option[]={
                   {'i', 0, PCRE_CASELESS, result}, // a=A
                   {'s', 0, PCRE_DOTALL, result}, // \n\n$ [default]
                   {'x', 0, PCRE_EXTENDED, result}, // whitespace in regex ignored
                   {'m', PCRE_DOTALL, PCRE_MULTILINE, result}, // ^aaa\n$^bbb\n$
                   {'g', 0, true, result+1}, // many rows
                   {0},
       };
           result[0]=PCRE_EXTRA | PCRE_DOTALL;
           result[1]=0;
   
       if(options) 
                   for(Regex_option *o=regex_option; o->key; o++) 
                           if(
                                   strchr(options, o->key) || 
                                   strchr(options, toupper(o->key))) {
                                   *(o->result)&=~o->clear;
                                   *(o->result)|=o->set;
                           }
   }
   
   /// @todo maybe need speedup: some option to remove pre/match/post string generation
   bool String::match(const unsigned char *pcre_tables,
                                      const String *aorigin,
                                      const String& regexp, 
                                      const String *options,
                                      Table **table,
                                      Row_action row_action, void *info,
                                      bool *was_global) const { 
   
           if(!regexp.size())
                   throw Exception(0, 0,
                           aorigin,
                           "regexp is empty");
           const char *pattern=regexp.cstr(UL_AS_IS);
           const char *errptr;
           int erroffset;
       int option_bits[2];  regex_options(options?options->cstr():0, option_bits);
           if(was_global)
                   *was_global=option_bits[1]!=0;
           pcre *code=pcre_compile(pattern, option_bits[0], 
                   &errptr, &erroffset,
                   pcre_tables);
   
           if(!code)
                   throw Exception(0, 0,
                           &regexp.mid(erroffset, regexp.size()),
                           "regular expression syntax error - %s", errptr);
           
           int info_substrings=pcre_info(code, 0, 0);
           if(info_substrings<0) {
                   pcre_free(code);
                   throw Exception(0, 0,
                           aorigin,
                           "pcre_info error (%d)", 
                                   info_substrings);
           }
   
           int startoffset=0;
           const char *subject=cstr(UL_AS_IS);
           int length=strlen(subject);
           int ovecsize;
           int *ovector=(int *)malloc(sizeof(int)*
                   (ovecsize=(1/*match*/+info_substrings)*3));
   
           { // create table
                   Array& columns=*NEW Array(pool());
                   columns+=string_pre_match_name;
                   columns+=string_match_name;
                   columns+=string_post_match_name;
                   for(int i=1; i<=info_substrings; i++) {
                           char *column=(char *)malloc(MAX_NUMBER);
                           snprintf(column, MAX_NUMBER, "%d", i);
                           columns+=NEW String(pool(), column); // .i column name
                   }
                   *table=NEW Table(pool(), aorigin, &columns);
           }
   
           int exec_option_bits=0;
           while(true) {
                   int exec_substrings=pcre_exec(code, 0,
                           subject, length, startoffset,
                           exec_option_bits, ovector, ovecsize);
                   
                   if(exec_substrings==PCRE_ERROR_NOMATCH) {
                           pcre_free(code);
                           row_action(**table, 0/*last time, no row*/, 0, 0, info);
                           return option_bits[1]!=0; // global=true+table, not global=false
                   }
   
                   if(exec_substrings<0) {
                           pcre_free(code);
                           throw Exception(0, 0,
                                   aorigin,
                                   "regular expression execute error (%d)", 
                                           exec_substrings);
                   }
   
                   Array& row=*NEW Array(pool());
                   row+=&mid(0, ovector[0]); // .prematch column value
                   row+=&mid(ovector[0], ovector[1]); // .match
                   row+=&mid(ovector[1], size()); // .postmatch
                   
                   for(int i=1; i<exec_substrings; i++) {
                           // -1:-1 case handled peacefully by mid() itself
                           row+=&mid(ovector[i*2+0], ovector[i*2+1]); // .i column value
                   }
                   
                   row_action(**table, &row, startoffset, ovector[0], info);
   
                   if(!option_bits[1] || startoffset==ovector[1]) { // not global | going to hang
                           pcre_free(code);
                           row_action(**table, 0/*last time, no row*/, 0, 0, info);
                         return true;                          return true;
         return false;                  }
                   startoffset=ovector[1];
   
   /*
                   if(option_bits[0] & PCRE_MULTILINE)
                           exec_option_bits|=PCRE_NOTBOL; // start of subject+startoffset not BOL
   */
           }
   }
   
   String& String::change_case(Pool& pool, const unsigned char *tables, 
                                                           Change_case_kind kind) const {
           String& result=*new(pool) String(pool);
   
           const unsigned char *a;
           const unsigned char *b;
           switch(kind) {
           case CC_UPPER:
                   a=tables+lcc_offset;
                   b=tables+fcc_offset;
                   break;
           case CC_LOWER:
                   a=tables+lcc_offset;
                   b=0;
                   break;
           default:
                   throw Exception(0, 0, 
                           this, 
                           "unknown change case kind #%d", 
                                   static_cast<int>(kind)); // never
                   a=b=0; // calm, compiler
                   break; // never
           }       
   
           const Chunk *chunk=&head; 
           do {
                   const Chunk::Row *row=chunk->rows;
                   for(size_t i=0; i<chunk->count; i++, row++) {
                           if(row==append_here)
                                   goto break2;
   
                           char *new_cstr=(char *)pool.malloc(row->item.size);
                           char *dest=new_cstr;
                           const char *src=row->item.ptr; 
                           for(int size=row->item.size; size--; src++) {
                                   unsigned char c=a[(unsigned char)*src];
                                   if(b)
                                           c=b[c];
   
                                   *dest++=(char)c;
                           }
                           
                           result.APPEND(new_cstr, row->item.size, 
                                   row->item.lang,
                                   row->item.origin.file, row->item.origin.line);
                   }
                   chunk=row->link;
           } while(chunk);
   break2:
   
           return result;
   }
   
   void String::join_chain(Pool& pool, 
                                              size_t& ai, const Chunk*& achunk, const Chunk::Row*& arow,
                                              Untaint_lang& joined_lang, const char *& joined_ptr, size_t& joined_size) const {
           joined_lang=arow->item.lang;
           
           // calc size
           joined_size=0;
           {
                   size_t start_i=ai;
                   const Chunk::Row *start_row=arow;
                   const Chunk *chunk=achunk;
                   do {
                           const Chunk::Row *row=start_row;
                           for(size_t i=start_i; i<chunk->count; i++, row++) {
                                   if(row==append_here)
                                           goto break21;
                                   
                                   if(row->item.lang==joined_lang)
                                           joined_size+=row->item.size;
                                   else
                                           break;
                           }
                           if(chunk=row->link) {
                                   start_i=0;
                                   start_row=chunk->rows;
                           } else
                                   break;
                   } while(true);
   break21:;
           }
   
           // if one row, return simply itself
           if(joined_size==arow->item.size) {
                   joined_ptr=arow->item.ptr;
                   ai++; arow++;
                   if(ai==achunk->count)
                           achunk=arow->link;              
           } else {
                   // join adjacent rows
                   char *ptr=(char *)pool.malloc(joined_size);
                   joined_ptr=ptr;
                   size_t start_i=ai;
                   const Chunk::Row *start_row=arow;
                   const Chunk *chunk=achunk;
                   size_t i;
                   const Chunk::Row *row;
                   do {
                           row=start_row;
                           for(i=start_i; i<chunk->count; i++, row++) {
                                   if(row==append_here)
                                           goto break22;
                                   
                                   if(row->item.lang==joined_lang) {
                                           memcpy(ptr, row->item.ptr, row->item.size);
                                           ptr+=row->item.size;
                                   } else
                                           break;
                           }
                           if(chunk=row->link) {
                                   start_i=0;
                                   start_row=chunk->rows;
                           } else
                                   break;
                   } while(true);
   break22:;
                   
                   // return joined rows
                   ai=i;
                   arow=row;
                   achunk=chunk;
           }
   }
   
   String& String::reconstruct(Pool& pool) const {
           //_asm int 3;
           String& result=*new(pool) String(pool);
           const Chunk *chunk=&head; 
           do {
                   const Chunk::Row *row=chunk->rows;
                   for(size_t i=0; i<chunk->count; ) {
                           if(row==append_here)
                                   goto break2;
   
                           Untaint_lang joined_lang;
                           const char *joined_ptr;
                           size_t joined_size;
                           join_chain(pool, i, chunk, row,
                                   joined_lang, joined_ptr, joined_size);
   
                           result.APPEND(joined_ptr, joined_size, 
                                   joined_lang,
                                   row->item.origin.file, row->item.origin.line);
                           if(!chunk)
                                   goto break2;
                   }
           } while(true);
   break2:
   
           return result;
   };
   
   String& String::replace_in_reconstructed(Pool& pool, Dictionary& dict) const {
           //_asm int 3;
           String& result=*new(pool) String(pool);
           const Chunk *chunk=&head; 
           do {
                   const Chunk::Row *row=chunk->rows;
                   for(size_t i=0; i<chunk->count; i++, row++) {
                           if(row==append_here)
                                   goto break2;
   
                           const char *src=row->item.ptr; 
                           size_t src_size=row->item.size;
                           char *new_cstr=(char *)pool.malloc((size_t)ceil(src_size*dict.max_ratio()));
                           char *dest=new_cstr;
                           while(src_size) {
                                   // there is a row where first column starts 'src'
                                   if(Table::Item *item=dict.first_that_starts(src, src_size)) {
                                           // get a=>b values
                                           const String& a=*static_cast<Array *>(item)->get_string(0);
                                           const String& b=*static_cast<Array *>(item)->get_string(1);
                                           // skip 'a' in 'src' && reduce work size
                                           src+=a.size();  src_size-=a.size();
                                           // write 'b' to 'dest' && skip 'b' in 'dest'
                                           b.store_to(dest);  dest+=b.size();
                                   } else {
                                           // write a char to b && reduce work size
                                           *dest++=*src++;  src_size--;
                                   }
                           }
   
                           result.APPEND(new_cstr, dest-new_cstr, 
                                   row->item.lang,
                                   row->item.origin.file, row->item.origin.line);
                   }
                   chunk=row->link;
           } while(chunk);
   
   break2:
           return result;
   }
   
   String& String::replace(Pool& pool, Dictionary& dict) const {
           return reconstruct(pool).replace_in_reconstructed(pool, dict);
   }
   
   double String::as_double() const { 
           double result;
           const char *cstr=this->cstr();
           char *error_pos;
           // 0xABC
           if(cstr[0]=='0')
                   if(cstr[1]=='x' || cstr[1]=='X')
                           result=(double)(unsigned long)strtol(cstr, &error_pos, 0);
                   else
                           result=(double)strtod(cstr+1/*skip leading 0*/, &error_pos);
           else
                   result=(double)strtod(cstr, &error_pos);
   
           if(*error_pos/*not EOS*/)
                   throw Exception(0, 0,
                           this,
                           "invalid number (double)");
   
           return result;
 }  }
   int String::as_int() const { 
           int result;
           const char *cstr=this->cstr();
           char *error_pos;
           // 0xABC
           if(cstr[0]=='0')
                   if(cstr[1]=='x' || cstr[1]=='X')
                           result=(int)(unsigned long)strtol(cstr, &error_pos, 0);
                   else
                           result=(int)strtol(cstr+1/*skip leading 0*/, &error_pos, 0);
           else
                   result=(int)strtol(cstr, &error_pos, 0);
   
           if(*error_pos/*not EOS*/)
                   throw Exception(0, 0,
                           this,
                           "invalid number (int)");
   
           return result;
   }
   
   /* @todo maybe network order worth spending some effort?
           don't bothering myself with network byte order,
           am not planning to be able to move resulting file across platforms
           for now
   */
   void String::serialize(size_t prolog_size, void *& buf, size_t& buf_size) const {
           buf_size=
                   prolog_size
                   +fused_rows*(sizeof(Untaint_lang)+sizeof(size_t))
                   +size();
           buf=malloc(buf_size);
           char *cur=(char *)buf+prolog_size;
   
           const Chunk *chunk=&head; 
           do {
                   const Chunk::Row *row=chunk->rows;
                   for(size_t i=0; i<chunk->count; i++) {
                           if(row==append_here)
                                   goto break2;
   
                           // lang
                           memcpy(cur, &row->item.lang, sizeof(Untaint_lang));
                           cur+=sizeof(Untaint_lang);
                           // size
                           memcpy(cur, &row->item.size, sizeof(size_t));
                           cur+=sizeof(size_t);
                           // bytes
                           memcpy(cur, row->item.ptr, row->item.size);
                           cur+=row->item.size;
   
                           row++;
                   }
                   chunk=row->link;
           } while(chunk);
   break2:
           ;
   }
   
   /* @todo maybe network order worth spending some effort?
           don't bothering myself with network byte order,
           am not planning to be able to move resulting file across platforms
           for now
   */
   #ifndef DOXYGEN
   struct Serialized_piece {
           String::Untaint_lang lang;
           size_t size;
           char ptr[1];
   };
   #endif
   
   void String::deserialize(size_t prolog_size, void *buf, size_t buf_size, const char *file) {
           char *cur=((char *)buf)+prolog_size;
           buf_size-=prolog_size;
   
           while(buf_size) {
                   Serialized_piece& p=*(Serialized_piece *)cur;
                   APPEND(p.ptr, p.size, p.lang, file, 0);
   
                   size_t piece_size=sizeof(p.lang)+sizeof(p.size)+p.size;
                   cur+=piece_size;
                   buf_size-=piece_size;
           }
   }
   

Removed from v.1.8  
changed lines
  Added in v.1.113


E-mail: