|
|
| version 1.26, 2001/09/21 14:46:09 | version 1.56, 2002/08/15 10:13:19 |
|---|---|
| Line 1 | Line 1 |
| /** @file | /** @file |
| Parser: pool class. | Parser: pool class. |
| Copyright (c) 2001 ArtLebedev Group (http://www.artlebedev.com) | Copyright (c) 2001, 2002 ArtLebedev Group (http://www.artlebedev.com) |
| Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru) | |
| Author: Alexander Petrosyan <paf@design.ru> (http://design.ru/paf) | |
| */ | */ |
| static const char *RCSId="$Id$"; | |
| static const char* IDENT_POOL_C="$Date$"; | |
| #include "pa_pool.h" | #include "pa_pool.h" |
| #include "pa_exception.h" | #include "pa_exception.h" |
| #include "pa_common.h" | #include "pa_common.h" |
| #include "pa_sapi.h" | |
| #ifdef XML | #include "pa_charset.h" |
| #include <PlatformSupport/DOMStringHelper.hpp> | |
| #include <util/PlatformUtils.hpp> | |
| #endif | |
| Pool::Pool(void *astorage) : | Pool::Pool(void *astorage) : |
| fstorage(astorage), fcontext(0), ftag(0), fexception(0) | fstorage(astorage), fcontext(0), |
| #ifdef XML | ftotal_allocated(0), ftotal_times(0), |
| , transcoder(0) | source_charset(0), client_charset(0) |
| #endif | |
| { | { |
| #ifdef XML | |
| charset=new(*this) String(*this, "UTF-8"); | |
| #endif | |
| } | } |
| Pool::~Pool() { | void *Pool::copy(const void *buf, const size_t size) { |
| #ifdef XML | void *result=malloc(size); |
| delete transcoder; | memcpy(result, buf, size); |
| #endif | return result; |
| } | |
| char *Pool::copy(const char *cstr) { | |
| if(cstr) { | |
| size_t size=strlen(cstr)+1; | |
| return (char *)copy(cstr, size); | |
| } | |
| return 0; | |
| } | } |
| void Pool::fail_alloc(size_t size) const { | void Pool::fail_alloc(size_t size) const { |
| fexception->_throw(0, 0, | SAPI::die("out of pool memory: failed to allocate %u bytes; " |
| 0, | "already allocated on pool: %u bytes in %u times", |
| "failed to allocate %u bytes", size); | size, |
| ftotal_allocated, ftotal_times); | |
| } | } |
| void Pool::fail_register_cleanup() const { | void Pool::fail_register_cleanup() const { |
| fexception->_throw(0, 0, | SAPI::die("failed to register cleanup"); |
| 0, | } |
| "failed to register cleanup"); | |
| void Pool::set_source_charset(Charset& acharset) { | |
| source_charset=&acharset; | |
| } | |
| Charset& Pool::get_source_charset() { | |
| if(!source_charset) | |
| throw Exception(0, | |
| 0, | |
| "no source charset defined yet"); | |
| return *source_charset; | |
| } | |
| void Pool::set_client_charset(Charset& acharset) { | |
| client_charset=&acharset; | |
| } | |
| Charset& Pool::get_client_charset() { | |
| if(!client_charset) | |
| throw Exception(0, | |
| 0, | |
| "no client charset defined yet"); | |
| return *client_charset; | |
| } | } |
| #ifdef XML | #ifdef XML |
| void Pool::set_charset(const String &new_charset) { | |
| if(new_charset!=*charset) { | const char *Pool::transcode_cstr(xmlChar *s) { |
| delete transcoder; transcoder=0; // flag "we need new transcoder" | return get_source_charset().transcode_cstr(s); |
| charset=&new_charset; // for this charset | |
| } | |
| } | } |
| void Pool::update_transcoder() { | String& Pool::transcode(xmlChar *s) { |
| if(transcoder) | return get_source_charset().transcode(s); |
| return; | } |
| XMLTransService::Codes resValue; | const char *Pool::transcode_cstr(GdomeDOMString *s) { |
| transcoder=XMLPlatformUtils::fgTransService->makeNewTranscoderFor(charset->cstr(), resValue, 60); | return get_source_charset().transcode_cstr(s); |
| if(!transcoder) | |
| THROW(0, 0, | |
| charset, | |
| "unsupported encoding"); | |
| } | |
| const char *Pool::transcode_cstr(const XalanDOMString& s) { | |
| update_transcoder(); | |
| const unsigned int len=s.size()*2; | |
| XMLByte* dest=(XMLByte *)malloc((len+1)*sizeof(XMLByte)); | |
| bool error=true; | |
| try { | |
| if(transcoder) { | |
| unsigned int charsEaten; | |
| unsigned int size=transcoder->transcodeTo( | |
| s.c_str(), s.length(), | |
| dest, len, | |
| charsEaten, | |
| XMLTranscoder::UnRep_RepChar //UnRep_Throw | |
| ); | |
| dest[size]=0; | |
| error=false; | |
| } | |
| } catch(...) { | |
| } | |
| if(error) { | |
| memset(dest, '?', s.size()); | |
| ((char *)dest)[s.size()]=0; | |
| } | |
| return (const char *)dest; | |
| } | } |
| String& Pool::transcode(const XalanDOMString& s) { | |
| return *new(*this) String(*this, transcode_cstr(s)); | String& Pool::transcode(GdomeDOMString *s) { |
| return get_source_charset().transcode(s); | |
| } | } |
| void Pool::_throw(const String *source, const XSLException& e) { | xmlChar *Pool::transcode_buf2xchar(const char *buf, size_t buf_size) { |
| if(e.getURI().empty()) | return get_source_charset().transcode_buf2xchar(buf, buf_size); |
| THROW(0, 0, | |
| source, | |
| "%s (%s)", | |
| transcode(e.getMessage()), // message for exception | |
| transcode(e.getType()) // type of exception | |
| ); | |
| else | |
| THROW(0, 0, | |
| source, | |
| "%s (%s) %s(%d:%d)'", | |
| transcode(e.getMessage()), // message for exception | |
| transcode(e.getType()), // type of exception | |
| transcode(e.getURI()), // URI for the associated document, if any | |
| e.getLineNumber(), // line number, or -1 if unknown | |
| e.getColumnNumber() // column number, or -1 if unknown | |
| ); | |
| } | } |
| GdomeDOMString_auto_ptr Pool::transcode(const String& s) { | |
| return get_source_charset().transcode(s); | |
| } | |
| #endif | #endif |