Diff for /parser3/src/main/pa_string.C between versions 1.78 and 1.117

version 1.78, 2001/04/23 13:38:31 version 1.117, 2001/10/29 16:56:31
Line 2 Line 2
         Parser: string class. @see untasize_t.C.          Parser: string class. @see untasize_t.C.
   
         Copyright (c) 2001 ArtLebedev Group (http://www.artlebedev.com)          Copyright (c) 2001 ArtLebedev Group (http://www.artlebedev.com)
   
         Author: Alexander Petrosyan <paf@design.ru> (http://design.ru/paf)          Author: Alexander Petrosyan <paf@design.ru> (http://design.ru/paf)
   
         $Id$          $Id$
Line 11 Line 10
 #include "pa_config_includes.h"  #include "pa_config_includes.h"
   
 #include "pcre.h"  #include "pcre.h"
   #include "internal.h"
   
 #include "pa_pool.h"  #include "pa_pool.h"
 #include "pa_string.h"  #include "pa_string.h"
Line 20 Line 20
 #include "pa_array.h"  #include "pa_array.h"
 #include "pa_globals.h"  #include "pa_globals.h"
 #include "pa_table.h"  #include "pa_table.h"
 #include "pa_threads.h"  #include "pa_dictionary.h"
   
 //#include "pa_sapi.h"  
   
 // String  
   
 String::String(Pool& apool, const char *src, size_t src_size, bool tainted) :  String::String(Pool& apool, const char *src, size_t src_size, bool tainted) :
         Pooled(apool) {          Pooled(apool),
           forigins_mode(false) {
         last_chunk=&head;          last_chunk=&head;
         head.count=CR_PREALLOCATED_COUNT;          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[head.count];          link_row=&head.rows[head.count];
         fused_rows=fsize=0;          fsize=0;
   
         if(src)          if(src)
                 if(tainted)                  if(tainted)
Line 42  String::String(Pool& apool, const char * Line 39  String::String(Pool& apool, const char *
                         APPEND_CLEAN(src, src_size, 0, 0);                          APPEND_CLEAN(src, src_size, 0, 0);
 }  }
   
 void String::expand() {  String::String(const String& src) :     
         size_t new_chunk_count=last_chunk->count+last_chunk->count*CR_GROW_PERCENT/100;          Pooled(src.pool()),
         last_chunk=static_cast<Chunk *>(          forigins_mode(false) {
                 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(const String& src) :     Pooled(src.pool()) {  
         head.count=CR_PREALLOCATED_COUNT;          head.count=CR_PREALLOCATED_COUNT;
                   
         size_t src_used_rows=src.fused_rows;          uint src_used_rows=src.used_rows();
         if(src_used_rows<=head.count) {          if(src_used_rows<=head.count) {
                 // all new rows fit size_to preallocated area                  // all new rows fit size_to preallocated area
                 size_t curr_chunk_rows=head.count;                  last_chunk=&head;
                   uint 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];
                 link_row=&head.rows[curr_chunk_rows];                  link_row=&head.rows[curr_chunk_rows];
Line 76  String::String(const String& src) : Pool Line 65  String::String(const String& src) : Pool
                 // 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 size_to new_chunk                  // remaining rows size_to new_chunk
                 size_t curr_chunk_rows=src_used_rows-head.count;                  uint curr_chunk_rows=src_used_rows-head.count;
                 Chunk *new_chunk=static_cast<Chunk *>(                  last_chunk=static_cast<Chunk *>(
                         malloc(sizeof(size_t)+sizeof(Chunk::Row)*curr_chunk_rows+sizeof(Chunk *)));                          malloc(sizeof(uint)+sizeof(Chunk::Row)*curr_chunk_rows+sizeof(Chunk *), 9));
                 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[new_chunk->count];                  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;
                 size_t rows_left_to_copy=new_chunk->count;                  uint rows_left_to_copy=last_chunk->count;
                 while(true) {                  while(true) {
                         size_t old_count=old_chunk->count;                          uint 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 106  String::String(const String& src) : Pool Line 95  String::String(const String& src) : Pool
                 }                  }
         }          }
         link_row->link=0;          link_row->link=0;
         fused_rows=src_used_rows;          src_used_rows;
         fsize=src.fsize;          fsize=src.fsize;
 }  }
   
   /// @todo not very optimal
   uint String::used_rows() const {
           uint result=0;
           const Chunk *chunk=&head; 
           do {
                   const Chunk::Row *row=chunk->rows;
                   for(uint i=0; i<chunk->count; i++, row++) {
                           if(row==append_here)
                                   goto break2;
   
                           result++;
                   }
                   chunk=row->link;
           } while(chunk);
   
   break2:
           return result;
   }
   void String::expand() {
           uint new_chunk_count=last_chunk->count+CR_GROW_COUNT;
           last_chunk=static_cast<Chunk *>(
                   malloc(sizeof(uint)+sizeof(Chunk::Row)*new_chunk_count+sizeof(Chunk *), 10));
           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) {  String& String::append(const String& src, Untaint_lang lang, bool forced) {
         const Chunk *chunk=&src.head;           const Chunk *chunk=&src.head; 
         do {          do {
                 const Chunk::Row *row=chunk->rows;                  const Chunk::Row *row=chunk->rows;
                 for(size_t i=0; i<chunk->count; i++, row++) {                  for(uint i=0; i<chunk->count; i++, row++) {
                         if(row==src.append_here)                          if(row==src.append_here)
                                 goto break2;                                  goto break2;
                                                   
                         APPEND(row->item.ptr, row->item.size,                           APPEND(row->item.ptr, row->item.size, 
                                 (lang!=UL_PASS_APPENDED && (row->item.lang==UL_TAINTED || forced))?lang:row->item.lang,                                  (lang!=UL_PASS_APPENDED && (row->item.lang==UL_TAINTED || forced))?lang:(Untaint_lang)row->item.lang,
                                 row->item.origin.file, row->item.origin.line);                                  row->item.origin.file, row->item.origin.line);
                 }                  }
                 chunk=row->link;                  chunk=row->link;
Line 146  String& String::real_append(STRING_APPEN Line 164  String& String::real_append(STRING_APPEN
         append_here->item.origin.file=file;          append_here->item.origin.file=file;
         append_here->item.origin.line=line;          append_here->item.origin.line=line;
 #endif  #endif
         append_here++; fused_rows++;          append_here++;
   
         return *this;          return *this;
 }  }
   
   char String::first_char() const {
           if(!fsize)
                   throw Exception(0, 0,
                           this,
                           "getting first char of empty string");
   
           return *head.rows[0].item.ptr;
   }
   
 uint String::hash_code() const {  uint String::hash_code() const {
         uint result=0;          uint result=0;
   
         const Chunk *chunk=&head;           const Chunk *chunk=&head; 
         do {          do {
                 const Chunk::Row *row=chunk->rows;                  const Chunk::Row *row=chunk->rows;
                 for(size_t i=0; i<chunk->count; i++) {                  for(uint i=0; i<chunk->count; i++) {
                         if(row==append_here)                          if(row==append_here)
                                 goto break2;                                  goto break2;
   
Line 184  int String::cmp(int& partial, const Stri Line 211  int String::cmp(int& partial, const Stri
         size_t b_offset=0;          size_t b_offset=0;
         Chunk::Row *a_end=append_here;          Chunk::Row *a_end=append_here;
         Chunk::Row *b_end=src.append_here;          Chunk::Row *b_end=src.append_here;
         size_t a_countdown=a_chunk->count;          uint a_countdown=a_chunk->count;
         size_t b_countdown=b_chunk->count;          uint b_countdown=b_chunk->count;
         bool a_break=false;          int result;
         bool b_break=false;  
         size_t result;  
         size_t pos=0;           size_t pos=0; 
         while(true) {  
                 a_break=a_row==a_end;  
                 b_break=b_row==b_end;  
                 if(a_break || b_break)  
                         break;  
   
           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(pos+a_row->item.size > this_offset) {
                         if(lang!=UL_UNSPECIFIED && a_row->item.lang!=lang)                           if(lang!=UL_UNSPECIFIED && a_row->item.lang!=lang) 
                                 return -1; // wrong lang -- bail out                                  return -1; // wrong lang -- bail out
Line 228  int String::cmp(int& partial, const Stri Line 251  int String::cmp(int& partial, const Stri
                                 pos+=a_row->item.size;                                  pos+=a_row->item.size;
                                 a_row++; a_countdown--; a_offset=0;                                  a_row++; a_countdown--; a_offset=0;
                         }                          }
                                                   if(b_break=b_row==b_end) {
                                   a_break=a_row==a_end;
                                   break;                  
                           }
                         if(!b_countdown) {                          if(!b_countdown) {
                                 b_chunk=b_row->link;                                  b_chunk=b_row->link;
                                 b_row=b_chunk->rows;                                  b_row=b_chunk->rows;
Line 240  int String::cmp(int& partial, const Stri Line 266  int String::cmp(int& partial, const Stri
                         a_row++; a_countdown--;                           a_row++; a_countdown--; 
                 }                  }
   
                   if(a_break=a_row==a_end) {
                           b_break=b_row==b_end;
                           break;
                   }
                 if(!a_countdown) {                  if(!a_countdown) {
                         a_chunk=a_row->link;                          a_chunk=a_row->link;
                         a_row=a_chunk->rows;                          a_row=a_chunk->rows;
Line 267  int String::cmp(int& partial, const char Line 297  int String::cmp(int& partial, const char
         size_t a_offset=this_offset;          size_t a_offset=this_offset;
         size_t b_offset=0;          size_t b_offset=0;
         Chunk::Row *a_end=append_here;          Chunk::Row *a_end=append_here;
         size_t a_countdown=a_chunk->count;          uint a_countdown=a_chunk->count;
         bool a_break=false;  
         bool b_break=false;  
         size_t pos=0;          size_t pos=0;
         while(true) {  
                 a_break=a_row==a_end;  
                 if(a_break || b_break)  
                         break;  
   
           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(pos+a_row->item.size > this_offset) {
                         if(lang!=UL_UNSPECIFIED && a_row->item.lang!=lang)                           if(lang!=UL_UNSPECIFIED && a_row->item.lang!=lang) 
                                 return -1; // wrong lang -- bail out                                  return -1; // wrong lang -- bail out
Line 285  int String::cmp(int& partial, const char Line 312  int String::cmp(int& partial, const char
                                 (b_size-b_offset);                                  (b_size-b_offset);
                                                   
                         if(size_diff==0) { // a has same size as b                          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,                                   if(int result=memcmp(a_row->item.ptr+a_offset, b_ptr+b_offset, 
                                         a_row->item.size-a_offset)!=0)                                          a_row->item.size-a_offset)!=0)
                                         return result;                                          return result;
                                 pos+=a_row->item.size;                                  pos+=a_row->item.size;
                                 a_row++; a_countdown--; a_offset=0;                                  a_row++; a_countdown--; a_offset=0;
                                 b_break=true;                                  b_break=true;
                         } else if (size_diff>0) { // a longer                          } else if (size_diff>0) { // a longer
                                 if(size_t result=memcmp(a_row->item.ptr+a_offset, b_ptr+b_offset,                                   if(int result=memcmp(a_row->item.ptr+a_offset, b_ptr+b_offset, 
                                         b_size-b_offset)!=0)                                          b_size-b_offset)!=0)
                                         return result;                                          return result;
                                 a_offset+=b_size-b_offset;                                  a_offset+=b_size-b_offset;
                                 b_break=true;                                  b_break=true;
                         } else { // b longer                          } else { // b longer
                                 if(size_t result=memcmp(a_row->item.ptr+a_offset, b_ptr+b_offset,                                   if(int result=memcmp(a_row->item.ptr+a_offset, b_ptr+b_offset, 
                                         a_row->item.size-a_offset)!=0)                                          a_row->item.size-a_offset)!=0)
                                         return result;                                          return result;
                                 b_offset+=a_row->item.size-a_offset;                                  b_offset+=a_row->item.size-a_offset;
Line 311  int String::cmp(int& partial, const char Line 338  int String::cmp(int& partial, const char
                         a_row++; a_countdown--;                           a_row++; a_countdown--; 
                 }                  }
   
                   a_break=a_row==a_end;
                   if(a_break || b_break)
                           break;
                 if(!a_countdown) {                  if(!a_countdown) {
                         a_chunk=a_row->link;                          a_chunk=a_row->link;
                         a_row=a_chunk->rows;                          a_row=a_chunk->rows;
Line 328  int String::cmp(int& partial, const char Line 358  int String::cmp(int& partial, const char
   
 #ifndef NO_STRING_ORIGIN  #ifndef NO_STRING_ORIGIN
 const Origin& String::origin() const {   const Origin& String::origin() const { 
         if(!fused_rows)          if(!fsize) {
                 THROW(0, 0,                   static const Origin empty_origin={"empty string"};
                         0,                  return empty_origin;
                         "String::origin() of empty string called");          }
                   
         // determining origin by last appended piece          // determining origin by last appended piece
         // because first one frequently constant.           // because first one frequently constant. 
         // ex: ^load[/file] "document_root" + "/file"          // ex: ^load[/file] "document_root" + "/file"
         return append_here[-1].item.origin;           // 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  #endif
   
 String& String::mid(size_t start, size_t finish) const {  String& String::mid(size_t start, size_t finish) const {
           String& result=*NEW String(pool());
   
         start=max(0, start);          start=max(0, start);
         finish=min(size(), finish);          finish=min(size(), finish);
         if(start==finish)          if(start==finish)
                 return *empty_string;                  return result;
   
         String& result=*NEW String(pool());  
   
         size_t pos=0;          size_t pos=0;
         const Chunk *chunk=&head;           const Chunk *chunk=&head; 
         do {          do {
                 const Chunk::Row *row=chunk->rows;                  const Chunk::Row *row=chunk->rows;
                 for(size_t i=0; i<chunk->count; pos+=row->item.size, i++, row++) {                  for(uint i=0; i<chunk->count; pos+=row->item.size, i++, row++) {
                         if(row==append_here)                          if(row==append_here)
                                 goto break2;                                  goto break2;
   
Line 364  String& String::mid(size_t start, size_t Line 398  String& String::mid(size_t start, size_t
                                 size_t size=finished?finish-pos:row->item.size;                                  size_t size=finished?finish-pos:row->item.size;
                                 result.APPEND(                                  result.APPEND(
                                         row->item.ptr+offset, size-offset,                                           row->item.ptr+offset, size-offset, 
                                         row->item.lang,                                          (Untaint_lang)row->item.lang,
                                         row->item.origin.file, row->item.origin.line);                                          row->item.origin.file, row->item.origin.line);
                                 if(finished)                                  if(finished)
                                         goto break2;                                          goto break2;
Line 379  break2: Line 413  break2:
 }  }
   
 int String::pos(const String& substr,   int String::pos(const String& substr, 
                                 size_t result, Untaint_lang lang) const {                                  int result, Untaint_lang lang) const {
         for(; result<size(); result++) {          for(; result<size(); result++) {
                 int partial; cmp(partial, substr, result, lang);                  int partial; cmp(partial, substr, result, lang);
                 if(                  if(
Line 392  int String::pos(const String& substr, Line 426  int String::pos(const String& substr,
 }  }
   
 int String::pos(const char *substr, size_t substr_size,   int String::pos(const char *substr, size_t substr_size, 
                                 size_t result, Untaint_lang lang) const {                                  int result, Untaint_lang lang) const {
         for(; result<size(); result++) {          for(; result<size(); result++) {
                 int partial; cmp(partial, substr, substr_size, result, lang);                  int partial; cmp(partial, substr, substr_size, result, lang);
                 if(                  if(
Line 463  static void regex_options(char *options, Line 497  static void regex_options(char *options,
                 int *result;                  int *result;
     } regex_option[]={      } regex_option[]={
                 {'i', 0, PCRE_CASELESS, result}, // a=A                  {'i', 0, PCRE_CASELESS, result}, // a=A
                 {'s', 0, PCRE_DOTALL, result}, // \n\n$                  {'s', 0, PCRE_DOTALL, result}, // \n\n$ [default]
                 {'x', 0, PCRE_EXTENDED, result}, // whitespace in regex ignored                  {'x', 0, PCRE_EXTENDED, result}, // whitespace in regex ignored
                 {'m', PCRE_DOTALL, PCRE_MULTILINE, result}, // ^aaa\n$^bbb\n$                  {'m', PCRE_DOTALL, PCRE_MULTILINE, result}, // ^aaa\n$^bbb\n$
                 {'g', 0, true, result+1}, // many rows                  {'g', 0, true, result+1}, // many rows
Line 482  static void regex_options(char *options, Line 516  static void regex_options(char *options,
                         }                          }
 }  }
   
 /**  /// @todo maybe need speedup: some option to remove pre/match/post string generation
         returns true if fills table.  
         table format is defined and fixed[can be used by others]:   
         @verbatim  
                 pre-match/match/post-match/1/2/3/...  
         @endverbatim  
 */  
 bool String::match(const unsigned char *pcre_tables,  bool String::match(const unsigned char *pcre_tables,
                                    const String *aorigin,                                     const String *aorigin,
                                    const String& regexp,                                      const String& regexp, 
                                    const String *options,                                     const String *options,
                                    Table **table,                                     Table **table,
                                    Row_action row_action, void *info) const {                                      Row_action row_action, void *info,
                                      bool *was_global) const { 
   
         if(!regexp.size())          if(!regexp.size())
                 THROW(0, 0,                  throw Exception(0, 0,
                         aorigin,                          aorigin,
                         "regexp is empty");                          "regexp is empty");
         const char *pattern=regexp.cstr(UL_AS_IS);          const char *pattern=regexp.cstr(UL_AS_IS);
         const char *errptr;          const char *errptr;
         int erroffset;          int erroffset;
     int option_bits[2];  regex_options(options?options->cstr():0, option_bits);      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],           pcre *code=pcre_compile(pattern, option_bits[0], 
                 &errptr, &erroffset,                  &errptr, &erroffset,
                 pcre_tables);                  pcre_tables);
   
         if(!code)          if(!code)
                 THROW(0, 0,                  throw Exception(0, 0,
                         &regexp.mid(erroffset, regexp.size()),                          &regexp.mid(erroffset, regexp.size()),
                         "regular expression syntax error - %s", errptr);                          "regular expression syntax error - %s", errptr);
                   
         int info_substrings=pcre_info(code, 0, 0);          int info_substrings=pcre_info(code, 0, 0);
         if(info_substrings<0) {          if(info_substrings<0) {
                 (*pcre_free)(code);                  pcre_free(code);
                 THROW(0, 0,                  throw Exception(0, 0,
                         aorigin,                          aorigin,
                         "pcre_info error (%d)",                           "pcre_info error (%d)", 
                                 info_substrings);                                  info_substrings);
Line 527  bool String::match(const unsigned char * Line 558  bool String::match(const unsigned char *
         int length=strlen(subject);          int length=strlen(subject);
         int ovecsize;          int ovecsize;
         int *ovector=(int *)malloc(sizeof(int)*          int *ovector=(int *)malloc(sizeof(int)*
                 (ovecsize=(1/*match*/+info_substrings)*3));                  (ovecsize=(1/*match*/+info_substrings)*3), 11);
   
         { // create table          { // create table
                 Array& columns=*NEW Array(pool());                  Array& columns=*NEW Array(pool());
Line 549  bool String::match(const unsigned char * Line 580  bool String::match(const unsigned char *
                         exec_option_bits, ovector, ovecsize);                          exec_option_bits, ovector, ovecsize);
                                   
                 if(exec_substrings==PCRE_ERROR_NOMATCH) {                  if(exec_substrings==PCRE_ERROR_NOMATCH) {
                         (*pcre_free)(code);                          pcre_free(code);
                         (*row_action)(**table, 0/*last time, no row*/, 0, 0, info);                          row_action(**table, 0/*last time, no row*/, 0, 0, info);
                         return option_bits[1]!=0; // global=true+table, not global=false                          return option_bits[1]!=0; // global=true+table, not global=false
                 }                  }
   
                 if(exec_substrings<0) {                  if(exec_substrings<0) {
                         (*pcre_free)(code);                          pcre_free(code);
                         THROW(0, 0,                          throw Exception(0, 0,
                                 aorigin,                                  aorigin,
                                 "regular expression execute error (%d)",                                   "regular expression execute error (%d)", 
                                         exec_substrings);                                          exec_substrings);
                 }                  }
   
                 Array& row=*NEW Array(pool());                  Array& row=*NEW Array(pool());
                 row+=&mid(0, ovector[0]); // .pre-match column value                  row+=&mid(0, ovector[0]); // .prematch column value
                 row+=&mid(ovector[0], ovector[1]); // .match                  row+=&mid(ovector[0], ovector[1]); // .match
                 row+=&mid(ovector[1], size()); // .post-match                  row+=&mid(ovector[1], size()); // .postmatch
                                   
                 for(int i=1; i<exec_substrings; i++) {                  for(int i=1; i<exec_substrings; i++) {
                         // -1:-1 case handled peacefully by mid() itself                          // -1:-1 case handled peacefully by mid() itself
                         row+=&mid(ovector[i*2+0], ovector[i*2+1]); // .i column value                          row+=&mid(ovector[i*2+0], ovector[i*2+1]); // .i column value
                 }                  }
                                   
                 (*row_action)(**table, &row, startoffset, ovector[0], info);                  row_action(**table, &row, startoffset, ovector[0], info);
   
                 if(!option_bits[1] || !(startoffset=ovector[1])) { // not global | going to hang                  if(!option_bits[1] || startoffset==ovector[1]) { // not global | going to hang
                         (*pcre_free)(code);                          pcre_free(code);
                         (*row_action)(**table, 0/*last time, no row*/, 0, 0, info);                          row_action(**table, 0/*last time, no row*/, 0, 0, info);
                         return true;                          return true;
                 }                  }
                   startoffset=ovector[1];
   
 /*  /*
                 if(option_bits[0] & PCRE_MULTILINE)                  if(option_bits[0] & PCRE_MULTILINE)
Line 586  bool String::match(const unsigned char * Line 618  bool String::match(const unsigned char *
 */  */
         }          }
 }  }
   
   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(uint i=0; i<chunk->count; i++, row++) {
                           if(row==append_here)
                                   goto break2;
   
                           char *new_cstr=(char *)pool.malloc(row->item.size, 12);
                           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, 
                                   (Untaint_lang)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, 
                                              uint& ai, const Chunk*& achunk, const Chunk::Row*& arow,
                                              Untaint_lang& joined_lang, const char *& joined_ptr, size_t& joined_size) const {
           joined_lang=(Untaint_lang)arow->item.lang;
           
           // calc size
           joined_size=0;
           {
                   uint start_i=ai;
                   const Chunk::Row *start_row=arow;
                   const Chunk *chunk=achunk;
                   do {
                           const Chunk::Row *row=start_row;
                           for(uint 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,13);
                   joined_ptr=ptr;
                   uint start_i=ai;
                   const Chunk::Row *start_row=arow;
                   const Chunk *chunk=achunk;
                   uint 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(uint 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(uint 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()), 14);
                           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, 
                                   (Untaint_lang)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;
           char buf[MAX_NUMBER];
           if(head.rows+1==append_here) {
                   int size=min(head.rows[0].item.size, MAX_NUMBER-1);
                   memcpy(buf, head.rows[0].item.ptr, size);
                   buf[size]=0;
                   cstr=buf;
           } else
                   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;
           char buf[MAX_NUMBER];
           if(head.rows+1==append_here) {
                   int size=min(head.rows[0].item.size, MAX_NUMBER-1);
                   memcpy(buf, head.rows[0].item.ptr, size);
                   buf[size]=0;
                   cstr=buf;
           } else
                   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
                   +used_rows()*(sizeof(Untaint_lang)+sizeof(size_t))
                   +size();
           buf=malloc(buf_size,15);
           char *cur=(char *)buf+prolog_size;
   
           const Chunk *chunk=&head; 
           do {
                   const Chunk::Row *row=chunk->rows;
                   for(uint 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.78  
changed lines
  Added in v.1.117


E-mail: