/* * CDE - Common Desktop Environment * * Copyright (c) 1993-2012, The Open Group. All rights reserved. * * These libraries and programs are free software; you can * redistribute them and/or modify them under the terms of the GNU * Lesser General Public License as published by the Free Software * Foundation; either version 2 of the License, or (at your option) * any later version. * * These libraries and programs are distributed in the hope that * they will be useful, but WITHOUT ANY WARRANTY; without even the * implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR * PURPOSE. See the GNU Lesser General Public License for more * details. * * You should have received a copy of the GNU Lesser General Public * License along with these librararies and programs; if not, write * to the Free Software Foundation, Inc., 51 Franklin Street, Fifth * Floor, Boston, MA 02110-1301 USA */ /* $XConsortium: Fixed2CodingSystem.C /main/1 1996/07/29 16:52:03 cde-hp $ */ // Copyright (c) 1994 James Clark // See the file COPYING for copying permission. // This uses a big endian byte order irrespective of host byte order. // Nothing special is done with FEFF/FFFE. #include "splib.h" #ifdef SP_MULTI_BYTE #include "Fixed2CodingSystem.h" #include "macros.h" #include #ifdef SP_NAMESPACE namespace SP_NAMESPACE { #endif class Fixed2Decoder : public Decoder { public: Fixed2Decoder(); size_t decode(Char *to, const char *from, size_t fromLen, const char **rest); Boolean convertOffset(unsigned long &offset) const; }; class Fixed2Encoder : public Encoder { public: Fixed2Encoder(); ~Fixed2Encoder(); void output(Char *, size_t, streambuf *); void output(const Char *, size_t, streambuf *); private: void allocBuf(size_t); char *buf_; size_t bufSize_; }; Decoder *Fixed2CodingSystem::makeDecoder() const { return new Fixed2Decoder; } Encoder *Fixed2CodingSystem::makeEncoder() const { return new Fixed2Encoder; } unsigned Fixed2CodingSystem::fixedBytesPerChar() const { return 2; } Fixed2Decoder::Fixed2Decoder() : Decoder(2) { } size_t Fixed2Decoder::decode(Char *to, const char *from, size_t fromLen, const char **rest) { #ifdef BIG_ENDIAN if (sizeof(Char) == 2 && from == (char *)to) { *rest = from + (fromLen & ~1); return fromLen/2; } #endif fromLen &= ~1; *rest = from + fromLen; for (size_t n = fromLen; n > 0; n -= 2) { *to++ = ((unsigned char)from[0] << 8) + (unsigned char)from[1]; from += 2; } return fromLen/2; } Boolean Fixed2Decoder::convertOffset(unsigned long &n) const { n *= 2; return true; } Fixed2Encoder::Fixed2Encoder() : buf_(0), bufSize_(0) { } Fixed2Encoder::~Fixed2Encoder() { delete [] buf_; } void Fixed2Encoder::allocBuf(size_t n) { if (bufSize_ < n) { delete [] buf_; buf_ = new char[bufSize_ = n]; } } // FIXME handle errors from streambuf::sputn void Fixed2Encoder::output(Char *s, size_t n, streambuf *sb) { #ifdef BIG_ENDIAN if (sizeof(Char) == 2) { sb->sputn((char *)s, n*2); return; } #endif ASSERT(sizeof(Char) >= 2); char *p = (char *)s; for (size_t i = 0; i < n; i++) { *p++ = (s[i] >> 8) & 0xff; *p++ = s[i] & 0xff; } sb->sputn((char *)s, n*2); } void Fixed2Encoder::output(const Char *s, size_t n, streambuf *sb) { #ifdef BIG_ENDIAN if (sizeof(Char) == 2) { sb->sputn((char *)s, n*2); return; } #endif allocBuf(n*2); for (size_t i = 0; i < n; i++) { buf_[i*2] = (s[i] >> 8) & 0xff; buf_[i*2 + 1] = s[i] & 0xff; } sb->sputn(buf_, n*2); } #ifdef SP_NAMESPACE } #endif #else /* not SP_MULTI_BYTE */ #ifndef __GNUG__ static char non_empty_translation_unit; // sigh #endif #endif /* not SP_MULTI_BYTE */