1999-08-16 21:50:52 +00:00
|
|
|
/*
|
2000-01-13 23:54:23 +00:00
|
|
|
*******************************************************************************
|
1999-08-16 21:50:52 +00:00
|
|
|
*
|
2000-01-13 23:54:23 +00:00
|
|
|
* Copyright (C) 1998-1999, International Business Machines
|
|
|
|
* Corporation and others. All Rights Reserved.
|
1999-08-16 21:50:52 +00:00
|
|
|
*
|
2000-01-13 23:54:23 +00:00
|
|
|
*******************************************************************************
|
1999-08-16 21:50:52 +00:00
|
|
|
*
|
|
|
|
* ucnv_err.c
|
|
|
|
* Implements error behaviour functions called by T_UConverter_{from,to}Unicode
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "uhash.h"
|
|
|
|
#include "ucmp8.h"
|
|
|
|
#include "ucmp16.h"
|
1999-12-28 23:39:02 +00:00
|
|
|
#include "unicode/ucnv_bld.h"
|
|
|
|
#include "unicode/ucnv_err.h"
|
1999-08-16 21:50:52 +00:00
|
|
|
#include "ucnv_cnv.h"
|
|
|
|
#include "cmemory.h"
|
1999-12-28 23:39:02 +00:00
|
|
|
#include "unicode/ucnv.h"
|
1999-08-16 21:50:52 +00:00
|
|
|
|
|
|
|
#define VALUE_STRING_LENGTH 32
|
|
|
|
/*Magic # 32 = 4(number of char in value string) * 8(max number of bytes per char for any converter) */
|
|
|
|
#define CODEPOINT_STRING_LENGTH 7
|
|
|
|
#define UNICODE_PERCENT_SIGN_CODEPOINT 0x0025
|
|
|
|
#define UNICODE_U_CODEPOINT 0x0055
|
|
|
|
#define UNICODE_X_CODEPOINT 0x0058
|
|
|
|
|
|
|
|
|
|
|
|
#define ToOffset(a) a<=9?(0x0030+a):(0x0030+a+7)
|
|
|
|
|
|
|
|
bool_t
|
1999-10-18 22:48:32 +00:00
|
|
|
CONVERSION_U_SUCCESS (UErrorCode err)
|
1999-08-16 21:50:52 +00:00
|
|
|
{
|
1999-10-07 00:07:53 +00:00
|
|
|
if ((err == U_INVALID_CHAR_FOUND) || (err == U_ILLEGAL_CHAR_FOUND)) return FALSE;
|
1999-08-16 21:50:52 +00:00
|
|
|
else return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*Takes a int32_t and fills in a UChar* string with that number "radix"-based
|
|
|
|
* and padded with "pad" zeroes
|
|
|
|
*/
|
|
|
|
static void itou (UChar * buffer, int32_t i, int32_t radix, int32_t pad)
|
|
|
|
{
|
|
|
|
int32_t length = 0;
|
|
|
|
int32_t num = 0;
|
|
|
|
int8_t digit;
|
|
|
|
int32_t j;
|
|
|
|
UChar temp;
|
|
|
|
|
2000-02-15 01:05:58 +00:00
|
|
|
while (i >= radix)
|
1999-08-16 21:50:52 +00:00
|
|
|
{
|
|
|
|
num = i / radix;
|
|
|
|
digit = (int8_t) (i - num * radix);
|
|
|
|
buffer[length++] = (UChar) (ToOffset (digit));
|
|
|
|
i = num;
|
|
|
|
}
|
|
|
|
|
|
|
|
buffer[length] = (UChar) (ToOffset (i));
|
|
|
|
|
|
|
|
while (length < pad) buffer[++length] = (UChar) 0x0030; /*zero padding */
|
|
|
|
buffer[length--] = (UChar) 0x0000;
|
|
|
|
|
|
|
|
/*Reverses the string */
|
|
|
|
for (j = 0; j < (pad / 2); j++)
|
|
|
|
{
|
|
|
|
temp = buffer[length - j];
|
|
|
|
buffer[length - j] = buffer[j];
|
|
|
|
buffer[j] = temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*Function Pointer STOPS at the ILLEGAL_SEQUENCE */
|
1999-10-18 22:48:32 +00:00
|
|
|
void UCNV_FROM_U_CALLBACK_STOP (UConverter * _this,
|
1999-08-16 21:50:52 +00:00
|
|
|
char **target,
|
|
|
|
const char *targetLimit,
|
|
|
|
const UChar ** source,
|
|
|
|
const UChar * sourceLimit,
|
|
|
|
int32_t *offsets,
|
|
|
|
bool_t flush,
|
|
|
|
UErrorCode * err)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*Function Pointer STOPS at the ILLEGAL_SEQUENCE */
|
1999-10-18 22:48:32 +00:00
|
|
|
void UCNV_TO_U_CALLBACK_STOP (UConverter * _this,
|
1999-08-16 21:50:52 +00:00
|
|
|
UChar ** target,
|
|
|
|
const UChar * targetLimit,
|
|
|
|
const char **source,
|
|
|
|
const char *sourceLimit,
|
|
|
|
int32_t *offsets,
|
|
|
|
bool_t flush,
|
|
|
|
UErrorCode * err)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
void UCNV_FROM_U_CALLBACK_SKIP (UConverter * _this,
|
1999-08-16 21:50:52 +00:00
|
|
|
char **target,
|
|
|
|
const char *targetLimit,
|
|
|
|
const UChar ** source,
|
|
|
|
const UChar * sourceLimit,
|
|
|
|
int32_t *offsets,
|
|
|
|
bool_t flush,
|
|
|
|
UErrorCode * err)
|
|
|
|
{
|
1999-10-18 22:48:32 +00:00
|
|
|
if (CONVERSION_U_SUCCESS (*err)) return;
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_ZERO_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
}
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
void UCNV_FROM_U_CALLBACK_SUBSTITUTE (UConverter * _this,
|
1999-08-16 21:50:52 +00:00
|
|
|
char **target,
|
|
|
|
const char *targetLimit,
|
|
|
|
const UChar ** source,
|
|
|
|
const UChar * sourceLimit,
|
|
|
|
int32_t *offsets,
|
|
|
|
bool_t flush,
|
|
|
|
UErrorCode * err)
|
|
|
|
{
|
|
|
|
char togo[5];
|
|
|
|
int32_t togoLen;
|
|
|
|
|
|
|
|
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
if (CONVERSION_U_SUCCESS (*err)) return;
|
1999-08-16 21:50:52 +00:00
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
/*In case we're dealing with a modal converter a la UCNV_EBCDIC_STATEFUL,
|
1999-08-16 21:50:52 +00:00
|
|
|
we need to make sure that the emitting of the substitution charater in the right mode*/
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy(togo, _this->subChar, togoLen = _this->subCharLen);
|
1999-10-18 22:48:32 +00:00
|
|
|
if (ucnv_getType(_this) == UCNV_EBCDIC_STATEFUL)
|
1999-08-16 21:50:52 +00:00
|
|
|
{
|
|
|
|
if ((_this->fromUnicodeStatus)&&(togoLen != 2))
|
|
|
|
{
|
|
|
|
togo[0] = UCNV_SI;
|
|
|
|
togo[1] = _this->subChar[0];
|
|
|
|
togo[2] = UCNV_SO;
|
|
|
|
togoLen = 3;
|
|
|
|
}
|
|
|
|
else if (!(_this->fromUnicodeStatus)&&(togoLen != 1))
|
|
|
|
{
|
|
|
|
togo[0] = UCNV_SO;
|
|
|
|
togo[1] = _this->subChar[0];
|
|
|
|
togo[2] = _this->subChar[1];
|
|
|
|
togo[3] = UCNV_SI;
|
|
|
|
togoLen = 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*if we have enough space on the output buffer we just copy
|
|
|
|
the subchar there and update the pointer */
|
|
|
|
if ((targetLimit - *target) >= togoLen)
|
|
|
|
{
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (*target, togo, togoLen);
|
1999-08-16 21:50:52 +00:00
|
|
|
*target += togoLen;
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_ZERO_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
if (offsets)
|
|
|
|
{
|
|
|
|
int i=0;
|
|
|
|
for (i=0;i<togoLen;i++) offsets[i]=0;
|
|
|
|
offsets += togoLen;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*if we don't have enough space on the output buffer
|
|
|
|
*we copy as much as we can to it, update that pointer.
|
|
|
|
*copy the rest in the internal buffer, and increase the
|
|
|
|
*length marker
|
|
|
|
*/
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (*target, togo, (targetLimit - *target));
|
1999-08-16 21:50:52 +00:00
|
|
|
if (offsets)
|
|
|
|
{
|
|
|
|
int i=0;
|
|
|
|
for (i=0;i<(targetLimit - *target);i++) offsets[i]=0;
|
|
|
|
offsets += (targetLimit - *target);
|
|
|
|
}
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (_this->charErrorBuffer + _this->charErrorBufferLength,
|
1999-08-16 21:50:52 +00:00
|
|
|
togo + (targetLimit - *target),
|
|
|
|
togoLen - (targetLimit - *target));
|
|
|
|
_this->charErrorBufferLength += togoLen - (targetLimit - *target);
|
|
|
|
*target += (targetLimit - *target);
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_INDEX_OUTOFBOUNDS_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/*uses itou to get a unicode escape sequence of the offensive sequence,
|
|
|
|
*uses a clean copy (resetted) of the converter, to convert that unicode
|
|
|
|
*escape sequence to the target codepage (if conversion failure happens then
|
|
|
|
*we revert to substituting with subchar)
|
|
|
|
*/
|
1999-10-18 22:48:32 +00:00
|
|
|
void UCNV_FROM_U_CALLBACK_ESCAPE (UConverter * _this,
|
1999-08-16 21:50:52 +00:00
|
|
|
char **target,
|
|
|
|
const char *targetLimit,
|
|
|
|
const UChar ** source,
|
|
|
|
const UChar * sourceLimit,
|
|
|
|
int32_t *offsets,
|
|
|
|
bool_t flush,
|
|
|
|
UErrorCode * err)
|
|
|
|
{
|
|
|
|
|
|
|
|
UChar valueString[VALUE_STRING_LENGTH];
|
|
|
|
int32_t valueStringLength = 0;
|
|
|
|
const UChar *mySource = *source;
|
|
|
|
UChar codepoint[CODEPOINT_STRING_LENGTH];
|
|
|
|
int32_t i = 0;
|
|
|
|
/*Makes a bitwise copy of the converter passwd in */
|
|
|
|
UConverter myConverter = *_this;
|
|
|
|
char myTarget[VALUE_STRING_LENGTH];
|
|
|
|
char *myTargetAlias = myTarget;
|
|
|
|
const UChar *myValueSource = NULL;
|
1999-10-07 00:07:53 +00:00
|
|
|
UErrorCode err2 = U_ZERO_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
uint32_t myFromUnicodeStatus = _this->fromUnicodeStatus;
|
|
|
|
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
if (CONVERSION_U_SUCCESS (*err)) return;
|
1999-08-16 21:50:52 +00:00
|
|
|
|
|
|
|
ucnv_reset (&myConverter);
|
|
|
|
myConverter.fromUnicodeStatus = myFromUnicodeStatus;
|
|
|
|
|
|
|
|
ucnv_setFromUCallBack (&myConverter,
|
1999-10-18 22:48:32 +00:00
|
|
|
(UConverterFromUCallback) UCNV_FROM_U_CALLBACK_STOP,
|
1999-08-16 21:50:52 +00:00
|
|
|
&err2);
|
1999-10-18 22:48:32 +00:00
|
|
|
if (U_FAILURE (err2))
|
1999-08-16 21:50:52 +00:00
|
|
|
{
|
|
|
|
*err = err2;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
codepoint[0] = (UChar) UNICODE_PERCENT_SIGN_CODEPOINT; /* adding % */
|
|
|
|
codepoint[1] = (UChar) UNICODE_U_CODEPOINT; /* adding U */
|
|
|
|
|
|
|
|
while (i < _this->invalidUCharLength)
|
|
|
|
{
|
|
|
|
itou (codepoint + 2, _this->invalidUCharBuffer[i++], 16, 4);
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (valueString + valueStringLength, codepoint, sizeof (UChar) * 6);
|
1999-08-16 21:50:52 +00:00
|
|
|
valueStringLength += CODEPOINT_STRING_LENGTH - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
myValueSource = valueString;
|
|
|
|
|
|
|
|
/*converts unicode escape sequence */
|
|
|
|
ucnv_fromUnicode (&myConverter,
|
|
|
|
&myTargetAlias,
|
|
|
|
myTargetAlias + VALUE_STRING_LENGTH,
|
|
|
|
&myValueSource,
|
|
|
|
myValueSource + CODEPOINT_STRING_LENGTH - 1,
|
|
|
|
NULL,
|
|
|
|
TRUE,
|
|
|
|
&err2);
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
if (U_FAILURE (err2))
|
1999-08-16 21:50:52 +00:00
|
|
|
{
|
1999-10-18 22:48:32 +00:00
|
|
|
UCNV_FROM_U_CALLBACK_SUBSTITUTE (_this,
|
1999-08-16 21:50:52 +00:00
|
|
|
target,
|
|
|
|
targetLimit,
|
|
|
|
source,
|
|
|
|
sourceLimit,
|
|
|
|
offsets,
|
|
|
|
flush,
|
|
|
|
err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
valueStringLength = myTargetAlias - myTarget;
|
|
|
|
|
|
|
|
/*if we have enough space on the output buffer we just copy
|
|
|
|
* the subchar there and update the pointer
|
|
|
|
*/
|
|
|
|
if ((targetLimit - *target) >= valueStringLength)
|
|
|
|
{
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (*target, myTarget, valueStringLength);
|
1999-08-16 21:50:52 +00:00
|
|
|
*target += valueStringLength;
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_ZERO_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
|
|
|
|
if (offsets)
|
|
|
|
{
|
|
|
|
int i=0;
|
|
|
|
for (i=0;i<valueStringLength;i++) offsets[i]=0;
|
|
|
|
offsets += valueStringLength;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*if we don't have enough space on the output buffer
|
|
|
|
*we copy as much as we can to it, update that pointer.
|
|
|
|
*copy the rest in the internal buffer, and increase the
|
|
|
|
*length marker
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (offsets)
|
|
|
|
{
|
|
|
|
int i=0;
|
|
|
|
for (i=0;i<(targetLimit - *target);i++) offsets[i]=0;
|
|
|
|
offsets += (targetLimit - *target);
|
|
|
|
}
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (*target, myTarget, (targetLimit - *target));
|
|
|
|
uprv_memcpy (_this->charErrorBuffer + _this->charErrorBufferLength,
|
1999-08-16 21:50:52 +00:00
|
|
|
myTarget + (targetLimit - *target),
|
|
|
|
valueStringLength - (targetLimit - *target));
|
|
|
|
_this->charErrorBufferLength += valueStringLength - (targetLimit - *target);
|
|
|
|
*target += (targetLimit - *target);
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_INDEX_OUTOFBOUNDS_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
void UCNV_TO_U_CALLBACK_SKIP (UConverter * _this,
|
1999-08-16 21:50:52 +00:00
|
|
|
UChar ** target,
|
|
|
|
const UChar * targetLimit,
|
|
|
|
const char **source,
|
|
|
|
const char *sourceLimit,
|
|
|
|
int32_t *offsets,
|
|
|
|
bool_t flush,
|
|
|
|
UErrorCode * err)
|
|
|
|
{
|
1999-10-18 22:48:32 +00:00
|
|
|
if (CONVERSION_U_SUCCESS (*err)) return;
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_ZERO_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
}
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
void UCNV_TO_U_CALLBACK_SUBSTITUTE (UConverter * _this,
|
1999-08-16 21:50:52 +00:00
|
|
|
UChar ** target,
|
|
|
|
const UChar * targetLimit,
|
|
|
|
const char **source,
|
|
|
|
const char *sourceLimit,
|
|
|
|
int32_t *offsets,
|
|
|
|
bool_t flush,
|
|
|
|
UErrorCode * err)
|
|
|
|
{
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
if (CONVERSION_U_SUCCESS (*err)) return;
|
1999-08-16 21:50:52 +00:00
|
|
|
|
|
|
|
if ((targetLimit - *target) >= 1)
|
|
|
|
{
|
|
|
|
**target = 0xFFFD;
|
|
|
|
(*target)++;
|
|
|
|
if (offsets) *offsets = 0;
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_ZERO_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_this->UCharErrorBuffer[_this->UCharErrorBufferLength] = 0xFFFD;
|
|
|
|
_this->UCharErrorBufferLength++;
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_INDEX_OUTOFBOUNDS_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/*uses itou to get a unicode escape sequence of the offensive sequence,
|
|
|
|
*and uses that as the substitution sequence
|
|
|
|
*/
|
1999-10-18 22:48:32 +00:00
|
|
|
void UCNV_TO_U_CALLBACK_ESCAPE (UConverter * _this,
|
1999-08-16 21:50:52 +00:00
|
|
|
UChar ** target,
|
|
|
|
const UChar * targetLimit,
|
|
|
|
const char **source,
|
|
|
|
const char *sourceLimit,
|
|
|
|
int32_t *offsets,
|
|
|
|
bool_t flush,
|
|
|
|
UErrorCode * err)
|
|
|
|
{
|
|
|
|
UChar uniValueString[VALUE_STRING_LENGTH];
|
|
|
|
int32_t valueStringLength = 0;
|
|
|
|
const unsigned char *mySource = (const unsigned char *) *source;
|
|
|
|
UChar codepoint[CODEPOINT_STRING_LENGTH];
|
|
|
|
int32_t j = 0, i = 0;
|
|
|
|
const int32_t* offsets_end = offsets +( targetLimit - *target);
|
|
|
|
|
1999-10-18 22:48:32 +00:00
|
|
|
if (CONVERSION_U_SUCCESS (*err)) return;
|
1999-08-16 21:50:52 +00:00
|
|
|
|
|
|
|
codepoint[0] = (UChar) UNICODE_PERCENT_SIGN_CODEPOINT; /* adding % */
|
|
|
|
codepoint[1] = (UChar) UNICODE_X_CODEPOINT; /* adding X */
|
|
|
|
|
|
|
|
while (i < _this->invalidCharLength)
|
|
|
|
{
|
|
|
|
itou (codepoint + 2, _this->invalidCharBuffer[i++], 16, 2);
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (uniValueString + valueStringLength, codepoint, sizeof (UChar) * 4);
|
1999-08-16 21:50:52 +00:00
|
|
|
valueStringLength += 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((targetLimit - *target) >= valueStringLength)
|
|
|
|
{
|
|
|
|
/*if we have enough space on the output buffer we just copy
|
|
|
|
* the subchar there and update the pointer
|
|
|
|
*/
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (*target, uniValueString, (sizeof (UChar)) * (valueStringLength));
|
1999-08-16 21:50:52 +00:00
|
|
|
if (offsets)
|
|
|
|
{
|
|
|
|
for (i = 0; i < valueStringLength; i++) offsets[i] = 0;
|
|
|
|
}
|
|
|
|
*target += valueStringLength;
|
|
|
|
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_ZERO_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*if we don't have enough space on the output buffer
|
|
|
|
*we copy as much as we can to it, update that pointer.
|
|
|
|
*copy the rest in the internal buffer, and increase the
|
|
|
|
*length marker
|
|
|
|
*/
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (*target, uniValueString, (sizeof (UChar)) * (targetLimit - *target));
|
1999-08-16 21:50:52 +00:00
|
|
|
if (offsets)
|
|
|
|
{
|
|
|
|
for (i = 0; i < (targetLimit - *target); i++) offsets[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-12-28 23:39:02 +00:00
|
|
|
uprv_memcpy (_this->UCharErrorBuffer,
|
1999-08-16 21:50:52 +00:00
|
|
|
uniValueString + (targetLimit - *target),
|
|
|
|
(sizeof (UChar)) * (valueStringLength - (targetLimit - *target)));
|
|
|
|
_this->UCharErrorBufferLength += valueStringLength - (targetLimit - *target);
|
|
|
|
*target += (targetLimit - *target);
|
1999-10-07 00:07:53 +00:00
|
|
|
*err = U_INDEX_OUTOFBOUNDS_ERROR;
|
1999-08-16 21:50:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|