summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--ChangeLog31
-rw-r--r--ace/CDR_Stream.cpp299
-rw-r--r--ace/CDR_Stream.h36
-rw-r--r--ace/CDR_Stream.i44
-rw-r--r--ace/OS.h2
-rw-r--r--ace/OS.i13
-rw-r--r--ace/ace_wchar.h8
-rw-r--r--ace/config-linux-common.h5
8 files changed, 329 insertions, 109 deletions
diff --git a/ChangeLog b/ChangeLog
index ef81e89b321..25e7584d988 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,34 @@
+Tue Mar 25 21:08:04 2003 Phil Mesnier <mesnier_p@ociweb.com>
+
+ * ace/CDR_Stream.cpp:
+ * ace/CDR_Stream.h:
+ * ace/CDR_Stream.i:
+
+ Added the value wchar_maxbytes_ to the ACE_OutputCDR which is
+ used to control the size of the wchar on the wire when no
+ translator is being used. This is because it is valid to use a
+ wchar codeset for which the maximum size is smaller than the
+ size of a wchar_t.
+
+ * ace/OS.h:
+
+ When defining the ACE-specific type WChar, added the existance
+ of the ACE_HAS_XPG4_MULTIBYTE_CHAR as a condition to allow WChar
+ to be an alias of wchar_t, rather than of ACE_UINT16.
+
+ * ace/OS.i:
+ * ace/config-linux-common.h:
+
+ As part of the wchar changes, it was discovered that older linux
+ platforms lack fgetwc() and ungetwc(), although all other wide
+ char related functions are supported. Added ACE_LACKS_FGETWC to
+ guard against this condition.
+
+ * ace/ace_wchar.h:
+
+ Rework the ifdef sieve to ensure that platforms such as solaris
+ that really do support wchar get initialized properly.
+
Tue Mar 25 09:01:00 2003 Ossama Othman <ossama@uci.edu>
* ace/Default_Constants.h (ACE_CONNECTOR_HANDLER_MAP_SIZE):
diff --git a/ace/CDR_Stream.cpp b/ace/CDR_Stream.cpp
index 2269dd038cd..8f1c9c2ec05 100644
--- a/ace/CDR_Stream.cpp
+++ b/ace/CDR_Stream.cpp
@@ -12,6 +12,8 @@ ACE_RCSID (ace,
// ****************************************************************
+int ACE_OutputCDR::wchar_maxbytes_ = sizeof (ACE_CDR::WChar);
+
ACE_OutputCDR::ACE_OutputCDR (size_t size,
int byte_order,
ACE_Allocator *buffer_allocator,
@@ -39,8 +41,7 @@ ACE_OutputCDR::ACE_OutputCDR (size_t size,
major_version_ (major_version),
minor_version_ (minor_version),
char_translator_ (0),
- wchar_translator_ (0),
- wchar_allowed_(1)
+ wchar_translator_ (0)
{
ACE_CDR::mb_align (&this->start_);
@@ -74,8 +75,7 @@ ACE_OutputCDR::ACE_OutputCDR (char *data, size_t size,
major_version_ (major_version),
minor_version_ (minor_version),
char_translator_ (0),
- wchar_translator_ (0),
- wchar_allowed_ (1)
+ wchar_translator_ (0)
{
// We cannot trust the buffer to be properly aligned
ACE_CDR::mb_align (&this->start_);
@@ -96,14 +96,19 @@ ACE_OutputCDR::ACE_OutputCDR (ACE_Message_Block *data,
major_version_ (major_version),
minor_version_ (minor_version),
char_translator_ (0),
- wchar_translator_ (0),
- wchar_allowed_ (1)
+ wchar_translator_ (0)
{
// We cannot trust the buffer to be properly aligned
ACE_CDR::mb_align (&this->start_);
this->current_ = &this->start_;
}
+/*static*/ void
+ACE_OutputCDR::wchar_maxbytes (int maxbytes)
+{
+ ACE_OutputCDR::wchar_maxbytes_ = maxbytes;
+}
+
int
ACE_OutputCDR::grow_and_adjust (size_t size,
size_t align,
@@ -169,31 +174,55 @@ ACE_OutputCDR::grow_and_adjust (size_t size,
ACE_CDR::Boolean
ACE_OutputCDR::write_wchar (ACE_CDR::WChar x)
{
- if (!this->wchar_allowed_)
+ if (this->wchar_translator_ != 0)
+ return (this->good_bit_ = this->wchar_translator_->write_wchar (*this, x));
+ if (ACE_OutputCDR::wchar_maxbytes_ == 0)
{
errno = EACCES;
return (this->good_bit_ = 0);
}
- if (this->wchar_translator_ != 0)
- return (this->good_bit_ = this->wchar_translator_->write_wchar (*this, x));
if (ACE_static_cast (ACE_CDR::Short, major_version_) == 1
&& ACE_static_cast (ACE_CDR::Short, minor_version_) == 2)
{
- ACE_CDR::Octet len = ACE_static_cast (ACE_CDR::Octet, sizeof(x));
+ ACE_CDR::Octet len = ACE_static_cast (ACE_CDR::Octet, ACE_OutputCDR::wchar_maxbytes_);
if (this->write_1 (&len))
- return this->write_octet_array (ACE_reinterpret_cast
- (const ACE_CDR::Octet*, &x),
- ACE_static_cast (ACE_CDR::ULong, len));
+ {
+ if (ACE_OutputCDR::wchar_maxbytes_ == sizeof(ACE_CDR::WChar))
+ return this->write_octet_array (ACE_reinterpret_cast
+ (const ACE_CDR::Octet*, &x),
+ ACE_static_cast (ACE_CDR::ULong, len));
+ else
+ if (ACE_OutputCDR::wchar_maxbytes_ == 2)
+ {
+ ACE_CDR::Short sx = ACE_static_cast(ACE_CDR::Short,x);
+ return this->write_octet_array(ACE_reinterpret_cast
+ (const ACE_CDR::Octet*, &sx),
+ ACE_static_cast (ACE_CDR::ULong, len));
+ }
+ else
+ {
+ ACE_CDR::Octet ox = ACE_static_cast(ACE_CDR::Octet,x);
+ return this->write_octet_array(ACE_reinterpret_cast
+ (const ACE_CDR::Octet*, &ox),
+ ACE_static_cast (ACE_CDR::ULong, len));
+ }
+ }
}
else if (ACE_static_cast (ACE_CDR::Short, minor_version_) == 0)
{ // wchar is not allowed with GIOP 1.0.
errno = EINVAL;
return (this->good_bit_ = 0);
}
- if (sizeof (ACE_CDR::WChar) == 2)
- return this->write_2 (ACE_reinterpret_cast (const ACE_CDR::UShort *, &x));
- else
+ if (ACE_OutputCDR::wchar_maxbytes_ == sizeof (ACE_CDR::WChar))
return this->write_4 (ACE_reinterpret_cast (const ACE_CDR::ULong *, &x));
+ else if (ACE_OutputCDR::wchar_maxbytes_ == 2)
+ {
+ ACE_CDR::Short sx = ACE_static_cast(ACE_CDR::Short,x);
+ return this->write_2 (ACE_reinterpret_cast (const ACE_CDR::UShort *,
+ &sx));
+ }
+ ACE_CDR::Octet ox = ACE_static_cast (ACE_CDR::Octet,x);
+ return this->write_1 (ACE_reinterpret_cast (const ACE_CDR::Octet *, &ox));
}
ACE_CDR::Boolean
@@ -241,24 +270,23 @@ ACE_OutputCDR::write_wstring (ACE_CDR::ULong len,
// i.e. normally the translator will be 0, but OTOH the code is
// smaller and should be better for the cache ;-) ;-)
// What do we do for GIOP 1.2???
- // Phil's answer->the translator writer's will have to deal.
- if (!this->wchar_allowed_)
+ if (this->wchar_translator_ != 0)
+ return this->wchar_translator_->write_wstring (*this, len, x);
+ if (ACE_OutputCDR::wchar_maxbytes_ == 0)
{
errno = EACCES;
return (this->good_bit_ = 0);
}
- if (this->wchar_translator_ != 0)
- return this->wchar_translator_->write_wstring (*this, len, x);
if (ACE_static_cast (ACE_CDR::Short, this->major_version_) == 1
- && ACE_static_cast (ACE_CDR::Short, this->minor_version_) == 2)
+ && ACE_static_cast (ACE_CDR::Short, this->minor_version_) == 2)
{
if (x != 0)
{
//In GIOP 1.2 the length field contains the number of bytes
//the wstring occupies rather than number of wchars
//Taking sizeof might not be a good way! This is a temporary fix.
- if (this->write_ulong (sizeof(ACE_CDR::WChar)*len))
+ if (this->write_ulong (ACE_OutputCDR::wchar_maxbytes_ * len))
return this->write_wchar_array (x, len);
}
else
@@ -474,6 +502,46 @@ ACE_OutputCDR::write_16 (const ACE_CDR::LongDouble *x)
}
ACE_CDR::Boolean
+ACE_OutputCDR::write_wchar_array_i (const ACE_CDR::WChar *x,
+ ACE_CDR::ULong length)
+{
+ if (length == 0)
+ return 1;
+ char* buf;
+ size_t align = (ACE_OutputCDR::wchar_maxbytes_ == 2) ?
+ ACE_CDR::SHORT_ALIGN :
+ ACE_CDR::OCTET_ALIGN;
+
+ if (this->adjust (ACE_OutputCDR::wchar_maxbytes_ * length, align, buf) == 0)
+ {
+ if (ACE_OutputCDR::wchar_maxbytes_ == 2)
+ {
+ ACE_CDR::UShort *sb = ACE_reinterpret_cast(ACE_CDR::UShort *, buf);
+ for (size_t i = 0; i < length; i++)
+#if !defined (ACE_ENABLE_SWAP_ON_WRITE)
+ sb[i] = ACE_static_cast (ACE_CDR::UShort, x[i]);
+#else
+ if (!this->do_byte_swap_)
+ sb[i] = ACE_static_cast (ACE_CDR::UShort, x[i]);;
+ else
+ {
+ ACE_CDR::UShort sx = ACE_static_cast (ACE_CDR::UShort, x[i]);
+ ACE_CDR::swap_2 (ACE_reinterpret_cast(char *,&sx),&buf[i*2]);
+ }
+#endif /* ACE_DISABLE_SWAP_ON_READ */
+ }
+ else
+ {
+ for (size_t i = 0; i < length; i++)
+ buf[i] = ACE_static_cast (ACE_CDR::Octet, x[i]);
+ }
+ return this->good_bit_;
+ }
+ return 0;
+}
+
+
+ACE_CDR::Boolean
ACE_OutputCDR::write_array (const void *x,
size_t size,
size_t align,
@@ -554,8 +622,7 @@ ACE_InputCDR::ACE_InputCDR (const char *buf,
major_version_ (major_version),
minor_version_ (minor_version),
char_translator_ (0),
- wchar_translator_ (0),
- wchar_allowed_ (1)
+ wchar_translator_ (0)
{
this->start_.wr_ptr (bufsiz);
}
@@ -570,8 +637,7 @@ ACE_InputCDR::ACE_InputCDR (size_t bufsiz,
major_version_ (major_version),
minor_version_ (minor_version),
char_translator_ (0),
- wchar_translator_ (0),
- wchar_allowed_ (1)
+ wchar_translator_ (0)
{
}
@@ -584,8 +650,8 @@ ACE_InputCDR::ACE_InputCDR (const ACE_Message_Block *data,
major_version_ (major_version),
minor_version_ (minor_version),
char_translator_ (0),
- wchar_translator_ (0),
- wchar_allowed_ (1)
+ wchar_translator_ (0)
+
{
this->reset (data, byte_order);
}
@@ -601,8 +667,8 @@ ACE_InputCDR::ACE_InputCDR (ACE_Data_Block *data,
major_version_ (major_version),
minor_version_ (minor_version),
char_translator_ (0),
- wchar_translator_ (0),
- wchar_allowed_ (1)
+ wchar_translator_ (0)
+
{
}
@@ -619,8 +685,8 @@ ACE_InputCDR::ACE_InputCDR (ACE_Data_Block *data,
major_version_ (major_version),
minor_version_ (minor_version),
char_translator_ (0),
- wchar_translator_ (0),
- wchar_allowed_ (1)
+ wchar_translator_ (0)
+
{
// Set the read pointer
this->start_.rd_ptr (rd_pos);
@@ -645,8 +711,8 @@ ACE_InputCDR::ACE_InputCDR (const ACE_InputCDR& rhs,
major_version_ (rhs.major_version_),
minor_version_ (rhs.minor_version_),
char_translator_ (rhs.char_translator_),
- wchar_translator_ (rhs.wchar_translator_),
- wchar_allowed_ (rhs.wchar_allowed_)
+ wchar_translator_ (rhs.wchar_translator_)
+
{
// Align the base pointer assuming that the incoming stream is also
// aligned the way we are aligned
@@ -675,8 +741,8 @@ ACE_InputCDR::ACE_InputCDR (const ACE_InputCDR& rhs,
major_version_ (rhs.major_version_),
minor_version_ (rhs.minor_version_),
char_translator_ (rhs.char_translator_),
- wchar_translator_ (rhs.wchar_translator_),
- wchar_allowed_ (rhs.wchar_allowed_)
+ wchar_translator_ (rhs.wchar_translator_)
+
{
// Align the base pointer assuming that the incoming stream is also
// aligned the way we are aligned
@@ -710,8 +776,7 @@ ACE_InputCDR::ACE_InputCDR (const ACE_InputCDR& rhs)
major_version_ (rhs.major_version_),
minor_version_ (rhs.minor_version_),
char_translator_ (rhs.char_translator_),
- wchar_translator_ (rhs.wchar_translator_),
- wchar_allowed_ (rhs.wchar_allowed_)
+ wchar_translator_ (rhs.wchar_translator_)
{
char *buf = ACE_ptr_align_binary (rhs.start_.base (),
ACE_CDR::MAX_ALIGNMENT);
@@ -729,8 +794,7 @@ ACE_InputCDR::ACE_InputCDR (ACE_InputCDR::Transfer_Contents x)
major_version_ (x.rhs_.major_version_),
minor_version_ (x.rhs_.minor_version_),
char_translator_ (x.rhs_.char_translator_),
- wchar_translator_ (x.rhs_.wchar_translator_),
- wchar_allowed_ (x.rhs_.wchar_allowed_)
+ wchar_translator_ (x.rhs_.wchar_translator_)
{
this->start_.rd_ptr (x.rhs_.start_.rd_ptr ());
@@ -751,7 +815,6 @@ ACE_InputCDR::operator= (const ACE_InputCDR& rhs)
this->do_byte_swap_ = rhs.do_byte_swap_;
this->good_bit_ = 1;
this->char_translator_ = rhs.char_translator_;
- this->wchar_allowed_ = rhs.wchar_allowed_;
this->major_version_ = rhs.major_version_;
this->minor_version_ = rhs.minor_version_;
}
@@ -778,8 +841,7 @@ ACE_InputCDR::ACE_InputCDR (const ACE_OutputCDR& rhs,
major_version_ (rhs.major_version_),
minor_version_ (rhs.minor_version_),
char_translator_ (rhs.char_translator_),
- wchar_translator_ (rhs.wchar_translator_),
- wchar_allowed_ (rhs.wchar_allowed_)
+ wchar_translator_ (rhs.wchar_translator_)
{
ACE_CDR::mb_align (&this->start_);
for (const ACE_Message_Block *i = rhs.begin ();
@@ -813,35 +875,92 @@ ACE_InputCDR::skip_wchar (void)
ACE_CDR::Boolean
ACE_InputCDR::read_wchar (ACE_CDR::WChar& x)
{
- if (!this->wchar_allowed_)
+ if (this->wchar_translator_ != 0)
+ {
+ this->good_bit_ = this->wchar_translator_->read_wchar (*this,x);
+ return this->good_bit_;
+ }
+ if (ACE_OutputCDR::wchar_maxbytes_ == 0)
{
errno = EACCES;
return (this->good_bit_ = 0);
}
- if (this->wchar_translator_ != 0)
+
+ if (ACE_OutputCDR::wchar_maxbytes_ == sizeof (ACE_CDR::WChar))
{
- this->good_bit_ = this->wchar_translator_->read_wchar (*this,x);
- return this->good_bit_;
+ if (ACE_static_cast (ACE_CDR::Short, major_version_) == 1
+ && ACE_static_cast (ACE_CDR::Short, minor_version_) == 2)
+ {
+ ACE_CDR::Octet len;
+
+ if (this->read_1 (&len))
+ return this->read_octet_array
+ (ACE_reinterpret_cast (ACE_CDR::Octet*, &x),
+ ACE_static_cast (ACE_CDR::ULong, len));
+ else
+ return (this->good_bit_ = 0);
+ }
+
+ if (sizeof (ACE_CDR::WChar) == 2)
+ return this->read_2 (ACE_reinterpret_cast (ACE_CDR::UShort *, &x));
+ else
+ return this->read_4 (ACE_reinterpret_cast (ACE_CDR::ULong *, &x));
}
+
if (ACE_static_cast (ACE_CDR::Short, major_version_) == 1
- && ACE_static_cast (ACE_CDR::Short, minor_version_) == 2)
+ && ACE_static_cast (ACE_CDR::Short, minor_version_) == 2)
{
ACE_CDR::Octet len;
if (this->read_1 (&len))
- return this->read_octet_array
- (ACE_reinterpret_cast (ACE_CDR::Octet*, &x),
- ACE_static_cast (ACE_CDR::ULong, len));
- else
- return (this->good_bit_ = 0);
+ {
+ if (len == 2)
+ {
+ ACE_CDR::Short sx;
+ if (this->read_octet_array
+ (ACE_reinterpret_cast (ACE_CDR::Octet*, &sx),
+ ACE_static_cast (ACE_CDR::ULong, len)))
+ {
+ x = ACE_static_cast(ACE_CDR::WChar, sx);
+ return 1;
+ }
+ }
+ else
+ {
+ ACE_CDR::Octet ox;
+ if (this->read_octet_array
+ (ACE_reinterpret_cast (ACE_CDR::Octet*, &ox),
+ ACE_static_cast (ACE_CDR::ULong, len)))
+ {
+ x = ACE_static_cast(ACE_CDR::WChar, ox);
+ return 1;
+ }
+ }
+ }
}
-
- if (sizeof (ACE_CDR::WChar) == 2)
- return this->read_2 (ACE_reinterpret_cast (ACE_CDR::UShort *,
- &x));
else
- return this->read_4 (ACE_reinterpret_cast (ACE_CDR::ULong *,
- &x));
+ {
+ if (ACE_OutputCDR::wchar_maxbytes_ == 2)
+ {
+ ACE_CDR::Short sx;
+ if (this->read_2 (ACE_reinterpret_cast (ACE_CDR::UShort *, &sx)))
+ {
+ x = ACE_static_cast(ACE_CDR::WChar, sx);
+ return 1;
+ }
+ }
+ else
+ {
+ ACE_CDR::Octet ox;
+ if (this->read_1 (&ox))
+ {
+ x = ACE_static_cast(ACE_CDR::WChar, ox);
+ return 1;
+ }
+
+ }
+ }
+ return (this->good_bit_ = 0);
}
ACE_CDR::Boolean
@@ -908,19 +1027,20 @@ ACE_InputCDR::read_wstring (ACE_CDR::WChar*& x)
// @@ This is a slight violation of "Optimize for the common case",
// i.e. normally the translator will be 0, but OTOH the code is
// smaller and should be better for the cache ;-) ;-)
- if (!this->wchar_allowed_)
- {
- errno = EACCES;
- return (this->good_bit_ = 0);
- }
if (this->wchar_translator_ != 0)
{
this->good_bit_ = this->wchar_translator_->read_wstring (*this, x);
return this->good_bit_;
}
+ if (ACE_OutputCDR::wchar_maxbytes_ == 0)
+ {
+ errno = EACCES;
+ return (this->good_bit_ = 0);
+ }
ACE_CDR::ULong len;
- this->read_ulong (len);
+ if (!this->read_ulong (len))
+ return 0;
// A check for the length being too great is done later in the
// call to read_char_array but we want to have it done before
@@ -931,7 +1051,7 @@ ACE_InputCDR::read_wstring (ACE_CDR::WChar*& x)
if (ACE_static_cast (ACE_CDR::Short, this->major_version_) == 1
&& ACE_static_cast (ACE_CDR::Short, this->minor_version_) == 2)
{
- len = len / sizeof (ACE_CDR::WChar);
+ len /= ACE_OutputCDR::wchar_maxbytes_;
//allocating one extra for the null character needed by applications
ACE_NEW_RETURN (x,
@@ -1024,6 +1144,47 @@ ACE_InputCDR::read_array (void* x,
}
ACE_CDR::Boolean
+ACE_InputCDR::read_wchar_array_i (ACE_CDR::WChar* x,
+ ACE_CDR::ULong length)
+{
+ if (length == 0)
+ return 1;
+ char* buf;
+ size_t align = (ACE_OutputCDR::wchar_maxbytes_ == 2) ?
+ ACE_CDR::SHORT_ALIGN :
+ ACE_CDR::OCTET_ALIGN;
+
+ if (this->adjust (ACE_OutputCDR::wchar_maxbytes_ * length, align, buf) == 0)
+ {
+ if (ACE_OutputCDR::wchar_maxbytes_ == 2)
+ {
+ ACE_CDR::UShort *sb = ACE_reinterpret_cast(ACE_CDR::UShort *, buf);
+ for (size_t i = 0; i < length; i++)
+#if defined (ACE_DISABLE_SWAP_ON_READ)
+ x[i] = ACE_static_cast (ACE_CDR::WChar, sb[i]);
+#else
+ if (!this->do_byte_swap_)
+ x[i] = ACE_static_cast (ACE_CDR::WChar, sb[i]);
+ else
+ {
+ ACE_CDR::UShort sx;
+ ACE_CDR::swap_2 (&buf[i*2], ACE_reinterpret_cast(char *,&sx));
+ x[i] = ACE_static_cast (ACE_CDR::WChar,sx);
+ }
+#endif /* ACE_DISABLE_SWAP_ON_READ */
+ }
+ else
+ {
+ for (size_t i = 0; i < length; i++)
+ x[i] = ACE_static_cast (ACE_CDR::WChar, buf[i]);
+ }
+ return this->good_bit_;
+ }
+ return 0;
+}
+
+
+ACE_CDR::Boolean
ACE_InputCDR::read_boolean_array (ACE_CDR::Boolean *x,
ACE_CDR::ULong length)
{
@@ -1076,6 +1237,7 @@ ACE_InputCDR::read_2 (ACE_CDR::UShort *x)
#endif /* ACE_DISABLE_SWAP_ON_READ */
return 1;
}
+ this->good_bit_ = 0;
return 0;
}
@@ -1095,6 +1257,7 @@ ACE_InputCDR::read_4 (ACE_CDR::ULong *x)
#endif /* ACE_DISABLE_SWAP_ON_READ */
return 1;
}
+ this->good_bit_ = 0;
return 0;
}
@@ -1145,6 +1308,7 @@ ACE_InputCDR::read_8 (ACE_CDR::ULongLong *x)
return 1;
}
+ this->good_bit_ = 0;
return 0;
}
@@ -1166,6 +1330,8 @@ ACE_InputCDR::read_16 (ACE_CDR::LongDouble *x)
#endif /* ACE_DISABLE_SWAP_ON_READ */
return 1;
}
+
+ this->good_bit_ = 0;
return 0;
}
@@ -1182,7 +1348,6 @@ ACE_InputCDR::skip_string (void)
}
this->good_bit_ = 0;
}
-
return 0;
}
diff --git a/ace/CDR_Stream.h b/ace/CDR_Stream.h
index 954b8f81b22..6d1884fb550 100644
--- a/ace/CDR_Stream.h
+++ b/ace/CDR_Stream.h
@@ -330,7 +330,7 @@ public:
/// Set the codeset translators.
void char_translator (ACE_Char_Codeset_Translator *);
void wchar_translator (ACE_WChar_Codeset_Translator *);
- void wchar_allowed (int );
+ static void wchar_maxbytes (int );
/**
* Return alignment of the wr_ptr(), with respect to the start of
@@ -399,6 +399,11 @@ private:
size_t align,
ACE_CDR::ULong length);
+
+ ACE_CDR::Boolean write_wchar_array_i (const ACE_CDR::WChar* x,
+ ACE_CDR::ULong length);
+
+
/**
* Grow the CDR stream. When it returns <buf> contains a pointer to
* memory in the CDR stream, with at least <size> bytes ahead of it
@@ -461,11 +466,15 @@ protected:
ACE_WChar_Codeset_Translator *wchar_translator_;
/**
- * There are some situations when it is an error to attempt wchar
- * i/o of any kind. This may be when using GIOP 1.0, or no valid
- * NCSW defined.
+ * Some wide char codesets may be defined with a maximum number
+ * of bytes that is smaller than the size of a wchar_t. This means
+ * that the CDR cannot simply memcpy a block of wchars to and from
+ * the stream, but must instead realign the bytes appropriately.
+ * In cases when wchar i/o is not allowed, such as with GIOP 1.0,
+ * or not having a native wchar codeset defined, the maxbytes is
+ * set to zero, indicating no wchar data is allowed.
*/
- int wchar_allowed_;
+ static int wchar_maxbytes_;
};
// ****************************************************************
@@ -827,7 +836,6 @@ public:
/// Set the codeset translators.
void char_translator (ACE_Char_Codeset_Translator *);
void wchar_translator (ACE_WChar_Codeset_Translator *);
- void wchar_allowed (int );
/**
* Returns (in <buf>) the next position in the buffer aligned to
@@ -871,13 +879,6 @@ protected:
ACE_Char_Codeset_Translator *char_translator_;
ACE_WChar_Codeset_Translator *wchar_translator_;
- /**
- * There are some situations when it is an error to attempt wchar
- * i/o of any kind. This may be when using GIOP 1.0, or no valid
- * NCSW defined.
- */
- int wchar_allowed_;
-
private:
ACE_CDR::Boolean read_1 (ACE_CDR::Octet *x);
ACE_CDR::Boolean read_2 (ACE_CDR::UShort *x);
@@ -909,6 +910,15 @@ private:
size_t align,
ACE_CDR::ULong length);
+ /**
+ * On those occasions when the native codeset for wchar is smaller than
+ * the size of a wchar_t, such as using UTF-16 with a 4-byte wchar_t, a
+ * special form of reading the array is needed. Actually, this should be
+ * a default translator.
+ */
+ ACE_CDR::Boolean read_wchar_array_i (ACE_CDR::WChar * x,
+ ACE_CDR::ULong length);
+
/// Move the rd_ptr ahead by <offset> bytes.
void rd_ptr (size_t offset);
diff --git a/ace/CDR_Stream.i b/ace/CDR_Stream.i
index a39ffe0542c..61b97320915 100644
--- a/ace/CDR_Stream.i
+++ b/ace/CDR_Stream.i
@@ -278,14 +278,21 @@ ACE_INLINE ACE_CDR::Boolean
ACE_OutputCDR::write_wchar_array (const ACE_CDR::WChar* x,
ACE_CDR::ULong length)
{
- if (this->wchar_translator_ == 0)
+ if (this->wchar_translator_)
+ return this->wchar_translator_->write_wchar_array (*this, x, length);
+ if (ACE_OutputCDR::wchar_maxbytes_ == 0)
+ {
+ errno = EACCES;
+ return (this->good_bit_ = 0);
+ }
+ if (ACE_OutputCDR::wchar_maxbytes_ == sizeof (ACE_CDR::WChar))
return this->write_array (x,
sizeof (ACE_CDR::WChar),
sizeof (ACE_CDR::WChar) == 2
? ACE_CDR::SHORT_ALIGN
: ACE_CDR::LONG_ALIGN,
length);
- return this->wchar_translator_->write_wchar_array (*this, x, length);
+ return this->write_wchar_array_i (x,length);
}
ACE_INLINE ACE_CDR::Boolean
@@ -528,13 +535,6 @@ ACE_INLINE void
ACE_OutputCDR::wchar_translator (ACE_WChar_Codeset_Translator * wctran)
{
this->wchar_translator_ = wctran;
- this->wchar_allowed_ = 1;
-}
-
-ACE_INLINE void
-ACE_OutputCDR::wchar_allowed (int allowed)
-{
- this->wchar_allowed_ = allowed;
}
// ****************************************************************
@@ -659,20 +659,22 @@ ACE_InputCDR::read_wchar_array (ACE_CDR::WChar* x,
{
// Make sure the length of the array isn't greater than the length of
// the stream.
- if (length * sizeof (ACE_CDR::WChar) > this->length())
+ if (length * ACE_OutputCDR::wchar_maxbytes_ > this->length())
{
this->good_bit_ = 0;
return 0;
}
- if (this->wchar_translator_ == 0)
- return this->read_array (x,
- sizeof (ACE_CDR::WChar),
- sizeof (ACE_CDR::WChar) == 2
- ? ACE_CDR::SHORT_ALIGN
- : ACE_CDR::LONG_ALIGN,
- length);
- return this->wchar_translator_->read_wchar_array (*this, x, length);
+ if (this->wchar_translator_ != 0)
+ return this->wchar_translator_->read_wchar_array (*this, x, length);
+ if (ACE_OutputCDR::wchar_maxbytes_ != sizeof (ACE_CDR::WChar))
+ return this->read_wchar_array_i (x, length);
+ return this->read_array (x,
+ sizeof (ACE_CDR::WChar),
+ sizeof (ACE_CDR::WChar) == 2
+ ? ACE_CDR::SHORT_ALIGN
+ : ACE_CDR::LONG_ALIGN,
+ length);
}
ACE_INLINE ACE_CDR::Boolean
@@ -1464,14 +1466,8 @@ ACE_INLINE void
ACE_InputCDR::wchar_translator (ACE_WChar_Codeset_Translator * wctran)
{
this->wchar_translator_ = wctran;
- this->wchar_allowed_ = 1;
}
-ACE_INLINE void
-ACE_InputCDR::wchar_allowed (int allowed)
-{
- this->wchar_allowed_ = allowed;
-}
// ****************************************************************
ACE_INLINE ACE_CDR::Boolean
diff --git a/ace/OS.h b/ace/OS.h
index 8f886a9ca8d..0bf671803cb 100644
--- a/ace/OS.h
+++ b/ace/OS.h
@@ -5776,7 +5776,7 @@ public:
// @@ (othman) IMHO, it is the lesser of two evils to use the
// correct type for the platform rather than (forcibly) assume
// that all wide characters are 16 bits.
-#ifdef ACE_HAS_WCHAR
+#if defined (ACE_HAS_WCHAR) || defined (ACE_HAS_XPG4_MULTIBYTE_CHAR)
typedef wchar_t WChar;
#else
typedef ACE_UINT16 WChar;
diff --git a/ace/OS.i b/ace/OS.i
index 80b23954c19..118e26ae371 100644
--- a/ace/OS.i
+++ b/ace/OS.i
@@ -7315,7 +7315,7 @@ ACE_OS::thr_setprio (ACE_hthread_t thr_id, int prio, int thr_policy)
if (result != 0)
result = -1;
- return result;
+ return result;
# endif /* ACE_HAS_PTHREADS_DRAFT4 */
# elif defined (ACE_HAS_STHREADS)
ACE_OSCALL_RETURN (ACE_ADAPT_RETVAL (::thr_setprio (thr_id, prio),
@@ -9953,13 +9953,24 @@ ACE_OS::clearerr (FILE* fp)
ACE_INLINE wint_t
ACE_OS::fgetwc (FILE* fp)
{
+#if defined (ACE_LACKS_FGETWC)
+ ACE_UNUSED_ARG (fp);
+ ACE_NOTSUP_RETURN (0);
+#else
ACE_OSCALL_RETURN (::fgetwc (fp), wint_t, WEOF);
+#endif /* ACE_LACKS_FGETWC */
}
ACE_INLINE wint_t
ACE_OS::ungetwc (wint_t c, FILE* fp)
{
+#if defined (ACE_LACKS_FGETWC)
+ ACE_UNUSED_ARG (c);
+ ACE_UNUSED_ARG (fp);
+ ACE_NOTSUP_RETURN (0);
+#else
ACE_OSCALL_RETURN (::ungetwc (c, fp), wint_t, WEOF);
+#endif /* ACE_LACKS_FGETWC */
}
#endif /* ACE_HAS_WCHAR */
diff --git a/ace/ace_wchar.h b/ace/ace_wchar.h
index 56d73bcc91a..08ccbd17859 100644
--- a/ace/ace_wchar.h
+++ b/ace/ace_wchar.h
@@ -49,6 +49,10 @@
#endif /* ACE_LEGACY_MODE */
+#if defined (ACE_HAS_XPG4_MULTIBYTE_CHAR)
+# include /**/ <wchar.h>
+#endif /* ACE_HAS_XPG4_MULTIBYPTE_CHAR */
+
#if defined (ACE_HAS_WCHAR)
# if defined (VXWORKS)
# include /**/ <types/vxTypes.h> /* For wchar_t */
@@ -64,9 +68,7 @@
# elif !defined (__BORLANDC__)
# include /**/ <wchar.h>
# endif /* ACE_HAS_STANDARD_CPP_LIBRARY */
-#elif defined (ACE_HAS_XPG4_MULTIBYTE_CHAR)
-# include /**/ <wchar.h>
-#endif
+#endif /* ACE_HAS_WCHAR */
#if defined (ACE_USES_STD_NAMESPACE_FOR_STDC_LIB) && \
(ACE_USES_STD_NAMESPACE_FOR_STDC_LIB != 0)
diff --git a/ace/config-linux-common.h b/ace/config-linux-common.h
index 3af388cff0a..7ea071e6cd0 100644
--- a/ace/config-linux-common.h
+++ b/ace/config-linux-common.h
@@ -198,6 +198,7 @@
#define ACE_LACKS_ITOW
#define ACE_LACKS_WCSICMP
#define ACE_LACKS_WCSNICMP
+#define ACE_LACKS_TOWLOWER
#if __GLIBC__ >= 2
# define ACE_HAS_3_PARAM_WCSTOK
@@ -254,6 +255,10 @@
// Platform lacks POSIX prototypes for certain System V functions
// like shared memory and message queues.
# define ACE_LACKS_SOME_POSIX_PROTOTYPES
+
+// glibc supports wchar, but lacks fgetwc and ungetwc
+#define ACE_LACKS_FGETWC
+
#endif
// glibc supports the mkstemp() function.