os/textandloc/charconvfw/charconv_fw/tools/convtool/convtool.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
* Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
* All rights reserved.
sl@0
     4
* This component and the accompanying materials are made available
sl@0
     5
* under the terms of "Eclipse Public License v1.0"
sl@0
     6
* which accompanies this distribution, and is available
sl@0
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
*
sl@0
     9
* Initial Contributors:
sl@0
    10
* Nokia Corporation - initial contribution.
sl@0
    11
*
sl@0
    12
* Contributors:
sl@0
    13
*
sl@0
    14
* Description: 
sl@0
    15
*
sl@0
    16
*/
sl@0
    17
sl@0
    18
sl@0
    19
#pragma warning (disable: 4514) // unreferenced inline/local function has been removed
sl@0
    20
sl@0
    21
#include <stdio.h>
sl@0
    22
#include <stdlib.h>
sl@0
    23
#include <string.h>
sl@0
    24
#include <locale.h>
sl@0
    25
#include <wchar.h>
sl@0
    26
#if defined(__VC32__)
sl@0
    27
#include <FCNTL.H>
sl@0
    28
#include <IO.H>
sl@0
    29
#endif
sl@0
    30
sl@0
    31
#undef BIG_ENDIAN
sl@0
    32
sl@0
    33
#if (defined(__MSVCRT__) || defined(_MSC_VER))
sl@0
    34
//#define _stricmp  _stricmp
sl@0
    35
//#define _strnicmp _strnicmp
sl@0
    36
#else // linux 
sl@0
    37
#define _stricmp  strcasecmp
sl@0
    38
#define _strnicmp strncasecmp
sl@0
    39
#endif
sl@0
    40
sl@0
    41
const int KVersionNumber=025;
sl@0
    42
const int KLargeNumber=1000000;
sl@0
    43
sl@0
    44
extern int Utf8ToUnicode(wchar_t* aUnicode, const char* aUtf8);
sl@0
    45
extern int UnicodeToUtf8(char* aUtf8, const wchar_t* aUnicode);
sl@0
    46
sl@0
    47
enum TByteOrder
sl@0
    48
	{
sl@0
    49
	EByteOrderUnspecified,
sl@0
    50
	EByteOrderBigEndian,
sl@0
    51
	EByteOrderLittleEndian,
sl@0
    52
#if defined(BIG_ENDIAN)
sl@0
    53
	EByteOrderNative=EByteOrderBigEndian,
sl@0
    54
	EByteOrderForeign=EByteOrderLittleEndian
sl@0
    55
#else
sl@0
    56
	EByteOrderNative=EByteOrderLittleEndian,
sl@0
    57
	EByteOrderForeign=EByteOrderBigEndian
sl@0
    58
#endif
sl@0
    59
	};
sl@0
    60
sl@0
    61
struct SBuffer
sl@0
    62
	{
sl@0
    63
	int iNumberOfBytes;
sl@0
    64
	void* iData;
sl@0
    65
	};
sl@0
    66
sl@0
    67
void PrintUsage(const char* aProgramName)
sl@0
    68
    {
sl@0
    69
	fprintf(stderr, "\nVersion %03d\n\nCharacter set conversion tool\nCopyright (c) 1999 Symbian Ltd\n\n", KVersionNumber);
sl@0
    70
	fprintf(stderr, "Usage:\n\n\t%s [<options>] <inputspec> <outputspec>\n\nwhere\n\n\t"
sl@0
    71
				"options    :=  [-big|-little][-byteordermark]\n\t"
sl@0
    72
				"inputspec  :=  -input=<format> [<input_file>]\n\t"
sl@0
    73
				"outputspec :=  -output=<format> [<output_file>]\n\t"
sl@0
    74
				"format     :=  unicode|1252|utf8|...\n\n", aProgramName);
sl@0
    75
	const char* localeData=setlocale(LC_ALL, "");
sl@0
    76
	while (*localeData!='.')
sl@0
    77
		{
sl@0
    78
		++localeData;
sl@0
    79
		}
sl@0
    80
	fprintf(stderr, "(The default encoding is currently \"%s\")\n\n", localeData+1);
sl@0
    81
	}
sl@0
    82
sl@0
    83
void Assert(int aCondition, const char* aErrorMessageFormat, const void* aExtraParameter1=NULL, const void* aExtraParameter2=NULL)
sl@0
    84
	{
sl@0
    85
	if (!aCondition)
sl@0
    86
		{
sl@0
    87
		char errorMessage[100];
sl@0
    88
		sprintf(errorMessage, aErrorMessageFormat, aExtraParameter1, aExtraParameter2);
sl@0
    89
		fprintf(stderr, "Error: %s\n", errorMessage);
sl@0
    90
		exit(1);
sl@0
    91
		}
sl@0
    92
	}
sl@0
    93
sl@0
    94
void PrintWarning(const char* aWarningMessage)
sl@0
    95
	{
sl@0
    96
	fprintf(stderr, "Warning: %s\n", aWarningMessage);
sl@0
    97
	}
sl@0
    98
sl@0
    99
int TryFileParameter(int aArgc, char* aArgv[], int& aArgIndex, const char* aInputOrOutput, const char*& aEncoding, FILE*& aFile, const char* aFileMode)
sl@0
   100
	{
sl@0
   101
	char prefix[100];
sl@0
   102
	strcpy(prefix, "-");
sl@0
   103
	strcat(prefix, aInputOrOutput);
sl@0
   104
	strcat(prefix, "=");
sl@0
   105
	int lengthOfPrefix=strlen(prefix);
sl@0
   106
	if (_strnicmp(aArgv[aArgIndex], prefix, lengthOfPrefix)==0)
sl@0
   107
		{
sl@0
   108
		Assert(aEncoding==NULL, "\"%s...\" is specified more than once", prefix);
sl@0
   109
		aEncoding=aArgv[aArgIndex]+lengthOfPrefix;
sl@0
   110
		++aArgIndex;
sl@0
   111
		if ((aArgIndex>=aArgc) || (aArgv[aArgIndex][0]=='-'))
sl@0
   112
			{
sl@0
   113
			--aArgIndex;
sl@0
   114
			}
sl@0
   115
		else
sl@0
   116
			{
sl@0
   117
			aFile=fopen(aArgv[aArgIndex], aFileMode);
sl@0
   118
			Assert(aFile!=NULL, "opening %s-file failed", aInputOrOutput);
sl@0
   119
			}
sl@0
   120
		return 1;
sl@0
   121
		}
sl@0
   122
	return 0;
sl@0
   123
	}
sl@0
   124
sl@0
   125
void ReadParameters(int aArgc, char* aArgv[], int& aOutputByteOrderMark, TByteOrder& aUnicodeByteOrder, const char*& aInputEncoding, const char*& aOutputEncoding, FILE*& aInputFile, FILE*& aOutputFile)
sl@0
   126
	{
sl@0
   127
	if ((aArgc<=1) || (_stricmp(aArgv[1], "?")==0) || (_stricmp(aArgv[1], "/?")==0))
sl@0
   128
		{
sl@0
   129
		PrintUsage(aArgv[0]);
sl@0
   130
		exit(0);
sl@0
   131
		}
sl@0
   132
	for (int i=1; i<aArgc; ++i) // start at index 1 to avoid the program name (which is the first parameter)
sl@0
   133
		{
sl@0
   134
		if (_stricmp(aArgv[i], "-byteordermark")==0)
sl@0
   135
			{
sl@0
   136
			Assert(!aOutputByteOrderMark, "\"-byteordermark\" is specified more than once");
sl@0
   137
			aOutputByteOrderMark=1;
sl@0
   138
			}
sl@0
   139
		else if (_stricmp(aArgv[i], "-big")==0)
sl@0
   140
			{
sl@0
   141
			Assert(aUnicodeByteOrder==EByteOrderUnspecified, "the byte order of unicode text (i.e. \"-big\"/\"-little\") is specified more than once");
sl@0
   142
			aUnicodeByteOrder=EByteOrderBigEndian;
sl@0
   143
			}
sl@0
   144
		else if (_stricmp(aArgv[i], "-little")==0)
sl@0
   145
			{
sl@0
   146
			Assert(aUnicodeByteOrder==EByteOrderUnspecified, "the byte order of unicode text (i.e. \"-big\"/\"-little\") is specified more than once");
sl@0
   147
			aUnicodeByteOrder=EByteOrderLittleEndian;
sl@0
   148
			}
sl@0
   149
		else
sl@0
   150
			{
sl@0
   151
			Assert(TryFileParameter(aArgc, aArgv, i, "input", aInputEncoding, aInputFile, "r") ||
sl@0
   152
				   TryFileParameter(aArgc, aArgv, i, "output", aOutputEncoding, aOutputFile, "w"), "bad parameter \"%s\"", aArgv[i]);
sl@0
   153
			}
sl@0
   154
		}
sl@0
   155
	Assert(aInputEncoding!=NULL, "no input encoding is specified");
sl@0
   156
	Assert(aOutputEncoding!=NULL, "no output encoding is specified");
sl@0
   157
	}
sl@0
   158
sl@0
   159
int ReadFromFileReturningNumberOfBytesRead(void* aBuffer, int aNumberOfBytesToRead, FILE* aInputFile)
sl@0
   160
	{
sl@0
   161
	int numberOfBytesRead=0;
sl@0
   162
	int numberOfBytesToReadThisTime=aNumberOfBytesToRead;
sl@0
   163
	for (;;)
sl@0
   164
		{
sl@0
   165
		for (;;)
sl@0
   166
			{
sl@0
   167
			const int remainingNumberOfBytesToRead=aNumberOfBytesToRead-numberOfBytesRead;
sl@0
   168
			if (numberOfBytesToReadThisTime>remainingNumberOfBytesToRead)
sl@0
   169
				{
sl@0
   170
				numberOfBytesToReadThisTime=remainingNumberOfBytesToRead;
sl@0
   171
				}
sl@0
   172
			const int numberOfBytesReadThisTime=fread(aBuffer, 1, numberOfBytesToReadThisTime, aInputFile);
sl@0
   173
			const int error=ferror(aInputFile);
sl@0
   174
			if (error==0)
sl@0
   175
				{
sl@0
   176
				aBuffer=((unsigned char*)aBuffer)+numberOfBytesReadThisTime;
sl@0
   177
				numberOfBytesRead+=numberOfBytesReadThisTime;
sl@0
   178
				Assert(numberOfBytesRead<=aNumberOfBytesToRead, "internal error (read too many bytes)");
sl@0
   179
				if ((numberOfBytesRead>=aNumberOfBytesToRead) || feof(aInputFile))
sl@0
   180
					{
sl@0
   181
					return numberOfBytesRead;
sl@0
   182
					}
sl@0
   183
				break;
sl@0
   184
				}
sl@0
   185
			numberOfBytesToReadThisTime/=2;
sl@0
   186
			Assert(numberOfBytesToReadThisTime>0, "reading from file failed with error number %d", (const void*)error);
sl@0
   187
			clearerr(aInputFile);
sl@0
   188
			}
sl@0
   189
		}
sl@0
   190
	}
sl@0
   191
sl@0
   192
void WriteToFile(const void* aBuffer, int aNumberOfBytesToWrite, FILE* aOutputFile)
sl@0
   193
	{
sl@0
   194
	const int numberOfBytesWritten=fwrite(aBuffer, 1, aNumberOfBytesToWrite, aOutputFile);
sl@0
   195
	Assert(numberOfBytesWritten==aNumberOfBytesToWrite, "only %d out of %d bytes could be written to file", (const void*)numberOfBytesWritten, (const void*)aNumberOfBytesToWrite);
sl@0
   196
	const int error=ferror(aOutputFile);
sl@0
   197
	Assert(error==0, "writing to file failed with error number %d", (const void*)error);
sl@0
   198
	}
sl@0
   199
sl@0
   200
void HandleByteOrderMarks(int aOutputByteOrderMark, TByteOrder& aUnicodeByteOrder, const char* aInputEncoding, const char* aOutputEncoding, FILE* aInputFile, FILE* aOutputFile)
sl@0
   201
	{
sl@0
   202
	if (_stricmp(aInputEncoding, "unicode")==0)
sl@0
   203
		{
sl@0
   204
		unsigned short firstUnicodeCharacter=0;
sl@0
   205
		const int numberOfBytesRead=ReadFromFileReturningNumberOfBytesRead((void*)&firstUnicodeCharacter, sizeof(unsigned short), aInputFile);
sl@0
   206
		TByteOrder byteOrderSpecifiedByByteOrderMark=EByteOrderUnspecified;
sl@0
   207
		if (numberOfBytesRead==sizeof(unsigned short))
sl@0
   208
			{
sl@0
   209
			switch (firstUnicodeCharacter)
sl@0
   210
				{
sl@0
   211
			case 0xfeff:
sl@0
   212
				byteOrderSpecifiedByByteOrderMark=EByteOrderNative;
sl@0
   213
				break;
sl@0
   214
			case 0xfffe:
sl@0
   215
				byteOrderSpecifiedByByteOrderMark=EByteOrderForeign;
sl@0
   216
				break;
sl@0
   217
			default:
sl@0
   218
				const int error=fseek(aInputFile, 0, SEEK_SET); // rewind to the start of the file
sl@0
   219
				Assert(error==0, "could not rewind to the start of the input file");
sl@0
   220
				break;
sl@0
   221
				}
sl@0
   222
			}
sl@0
   223
		if (byteOrderSpecifiedByByteOrderMark!=EByteOrderUnspecified)
sl@0
   224
			{
sl@0
   225
			if ((aUnicodeByteOrder!=EByteOrderUnspecified) && (byteOrderSpecifiedByByteOrderMark!=aUnicodeByteOrder))
sl@0
   226
				{
sl@0
   227
				PrintWarning("the byte order specified by the byte-order mark in the unicode input is different from the byte order specified by the parameter - taking the byte-order specified by the byte-order mark in the unicode input");
sl@0
   228
				}
sl@0
   229
			aUnicodeByteOrder=byteOrderSpecifiedByByteOrderMark;
sl@0
   230
			}
sl@0
   231
		}
sl@0
   232
	if (aOutputByteOrderMark)
sl@0
   233
		{
sl@0
   234
		if (_stricmp(aOutputEncoding, "unicode")!=0)
sl@0
   235
			{
sl@0
   236
			PrintWarning("\"-byteordermark\" is only relevant for unicode output");
sl@0
   237
			}
sl@0
   238
		else
sl@0
   239
			{
sl@0
   240
			Assert(aUnicodeByteOrder!=EByteOrderUnspecified, "the byte order must be specified if a byte-order mark is to be added to the unicode output");
sl@0
   241
			unsigned short firstUnicodeCharacter=(unsigned short)((aUnicodeByteOrder==EByteOrderNative)? 0xfeff: 0xfffe);
sl@0
   242
			WriteToFile((const void*)&firstUnicodeCharacter, sizeof(unsigned short), aOutputFile);
sl@0
   243
			}
sl@0
   244
		}
sl@0
   245
	}
sl@0
   246
sl@0
   247
void ObeyRequiredByteOrderIfUnicode(TByteOrder& aUnicodeByteOrder, const char* aEncoding, SBuffer& aBuffer)
sl@0
   248
	{
sl@0
   249
	if (_stricmp(aEncoding, "unicode")==0)
sl@0
   250
		{
sl@0
   251
		Assert(aBuffer.iNumberOfBytes%sizeof(wchar_t)==0, "internal error (bad number of bytes in unicode buffer)");
sl@0
   252
		if (aUnicodeByteOrder==EByteOrderUnspecified)
sl@0
   253
			{
sl@0
   254
			PrintWarning("the byte order of unicode text is unspecified - defaulting to little endian");
sl@0
   255
			aUnicodeByteOrder=EByteOrderLittleEndian;
sl@0
   256
			}
sl@0
   257
		if (aUnicodeByteOrder==EByteOrderForeign)
sl@0
   258
			{
sl@0
   259
			for (unsigned char* bytePointer=((unsigned char*)aBuffer.iData)+(aBuffer.iNumberOfBytes-sizeof(wchar_t)); bytePointer>=aBuffer.iData; bytePointer-=sizeof(wchar_t))
sl@0
   260
				{
sl@0
   261
				unsigned char temp=*bytePointer;
sl@0
   262
				*bytePointer=*(bytePointer+1);
sl@0
   263
				*(bytePointer+1)=temp;
sl@0
   264
				}
sl@0
   265
			}
sl@0
   266
		}
sl@0
   267
	}
sl@0
   268
sl@0
   269
int OtherToUnicode(const char* aInputEncoding, wchar_t* aUnicode, const char* aOther)
sl@0
   270
// if the output parameter is NULL, it returns the precise size of the would-be output parameter (in terms of number of "wchar_t"s) excluding any trailing '\0', otherwise it returns 0
sl@0
   271
	{
sl@0
   272
	if (_stricmp(aInputEncoding, "utf8")==0)
sl@0
   273
		{
sl@0
   274
		return Utf8ToUnicode(aUnicode, aOther);
sl@0
   275
		}
sl@0
   276
	char localeData[100];
sl@0
   277
	strcpy(localeData, ".");
sl@0
   278
	strcat(localeData, aInputEncoding);
sl@0
   279
	Assert(setlocale(LC_ALL, localeData)!=NULL, "could not convert from encoding \"%s\"", aInputEncoding);
sl@0
   280
	return mbstowcs(aUnicode, aOther, KLargeNumber);
sl@0
   281
	}
sl@0
   282
sl@0
   283
int UnicodeToOther(const char* aOutputEncoding, char* aOther, const wchar_t* aUnicode)
sl@0
   284
// if the output parameter is NULL, it returns the precise size of the would-be output parameter (in terms of number of "char"s) excluding any trailing '\0', otherwise it returns 0
sl@0
   285
	{
sl@0
   286
	if (_stricmp(aOutputEncoding, "utf8")==0)
sl@0
   287
		{
sl@0
   288
		return UnicodeToUtf8(aOther, aUnicode);
sl@0
   289
		}
sl@0
   290
	char localeData[100];
sl@0
   291
	strcpy(localeData, ".");
sl@0
   292
	strcat(localeData, aOutputEncoding);
sl@0
   293
	Assert(setlocale(LC_ALL, localeData)!=NULL, "could not convert to encoding \"%s\"", aOutputEncoding);
sl@0
   294
	return wcstombs(aOther, aUnicode, KLargeNumber);
sl@0
   295
	}
sl@0
   296
sl@0
   297
void DoConversion(TByteOrder& aUnicodeByteOrder, const char* aInputEncoding, const char* aOutputEncoding, FILE* aInputFile, FILE* aOutputFile)
sl@0
   298
	{
sl@0
   299
	SBuffer arrayOfBuffers[3];
sl@0
   300
	arrayOfBuffers[0].iNumberOfBytes=0;
sl@0
   301
	arrayOfBuffers[0].iData=malloc(KLargeNumber+2); // +2 for the 2 '\0' bytes appended to the data read from file
sl@0
   302
	Assert(arrayOfBuffers[0].iData!=NULL, "cannot allocate enough memory");
sl@0
   303
	arrayOfBuffers[1].iNumberOfBytes=0;
sl@0
   304
	arrayOfBuffers[1].iData=NULL;
sl@0
   305
	arrayOfBuffers[2].iNumberOfBytes=0;
sl@0
   306
	arrayOfBuffers[2].iData=NULL;
sl@0
   307
	SBuffer* currentBuffer=arrayOfBuffers;
sl@0
   308
	currentBuffer->iNumberOfBytes=ReadFromFileReturningNumberOfBytesRead(currentBuffer->iData, KLargeNumber, aInputFile);
sl@0
   309
	// append 2 '\0' bytes at the end of the buffer read from file (2 in case it is unicode)
sl@0
   310
	((char*)currentBuffer->iData)[currentBuffer->iNumberOfBytes]='\0';
sl@0
   311
	((char*)currentBuffer->iData)[currentBuffer->iNumberOfBytes+1]='\0';
sl@0
   312
	ObeyRequiredByteOrderIfUnicode(aUnicodeByteOrder, aInputEncoding, *currentBuffer);
sl@0
   313
	// if the input and output encodings are different, convert from one to the other (via unicode if neither is itself unicode)
sl@0
   314
	if (_stricmp(aInputEncoding, aOutputEncoding)!=0)
sl@0
   315
		{
sl@0
   316
		if (_stricmp(aInputEncoding, "unicode")!=0)
sl@0
   317
			{
sl@0
   318
			SBuffer* nextBuffer=currentBuffer+1;
sl@0
   319
			nextBuffer->iNumberOfBytes=sizeof(wchar_t)*OtherToUnicode(aInputEncoding, NULL, (const char*)currentBuffer->iData);
sl@0
   320
			Assert(nextBuffer->iNumberOfBytes>=0, "invalid multi-byte character encountered");
sl@0
   321
			nextBuffer->iData=malloc(nextBuffer->iNumberOfBytes+sizeof(wchar_t)); // "+sizeof(wchar_t)" for terminating '\0'
sl@0
   322
			Assert(nextBuffer->iData!=NULL, "cannot allocate enough memory");
sl@0
   323
			OtherToUnicode(aInputEncoding, (wchar_t*)nextBuffer->iData, (const char*)currentBuffer->iData);
sl@0
   324
			currentBuffer=nextBuffer;
sl@0
   325
			}
sl@0
   326
		if (_stricmp(aOutputEncoding, "unicode")!=0)
sl@0
   327
			{
sl@0
   328
			SBuffer* nextBuffer=currentBuffer+1;
sl@0
   329
			nextBuffer->iNumberOfBytes=sizeof(char)*UnicodeToOther(aOutputEncoding, NULL, (const wchar_t*)currentBuffer->iData);
sl@0
   330
			Assert(nextBuffer->iNumberOfBytes>=0, "unconvertible unicode character encountered");
sl@0
   331
			nextBuffer->iData=malloc(nextBuffer->iNumberOfBytes+sizeof(char)); // "+sizeof(char)" for terminating '\0'
sl@0
   332
			Assert(nextBuffer->iData!=NULL, "cannot allocate enough memory");
sl@0
   333
			UnicodeToOther(aOutputEncoding, (char*)nextBuffer->iData, (const wchar_t*)currentBuffer->iData);
sl@0
   334
			currentBuffer=nextBuffer;
sl@0
   335
			}
sl@0
   336
		}
sl@0
   337
	ObeyRequiredByteOrderIfUnicode(aUnicodeByteOrder, aOutputEncoding, *currentBuffer);
sl@0
   338
	WriteToFile((const void*)currentBuffer->iData, currentBuffer->iNumberOfBytes, aOutputFile);
sl@0
   339
	free(arrayOfBuffers[0].iData);
sl@0
   340
	free(arrayOfBuffers[1].iData);
sl@0
   341
	free(arrayOfBuffers[2].iData);
sl@0
   342
	}
sl@0
   343
sl@0
   344
void FlushAndCloseFiles(FILE* aInputFile, FILE* aOutputFile)
sl@0
   345
	{
sl@0
   346
	Assert(fflush(aOutputFile)==0, "flushing output-file failed");
sl@0
   347
	if (aInputFile!=stdin)
sl@0
   348
		{
sl@0
   349
		Assert(fclose(aInputFile)==0, "closing input-file failed");
sl@0
   350
		}
sl@0
   351
	if (aOutputFile!=stdout)
sl@0
   352
		{
sl@0
   353
		Assert(fclose(aOutputFile)==0, "closing output-file failed");
sl@0
   354
		}
sl@0
   355
	}
sl@0
   356
sl@0
   357
int main(int aArgc, char* aArgv[])
sl@0
   358
	{
sl@0
   359
	int outputByteOrderMark=0;
sl@0
   360
	TByteOrder unicodeByteOrder=EByteOrderUnspecified;
sl@0
   361
	const char* inputEncoding=NULL;
sl@0
   362
	const char* outputEncoding=NULL;
sl@0
   363
	FILE* inputFile=stdin;
sl@0
   364
	FILE* outputFile=stdout;
sl@0
   365
	ReadParameters(aArgc, aArgv, outputByteOrderMark, unicodeByteOrder, inputEncoding, outputEncoding, inputFile, outputFile);
sl@0
   366
#if defined(__VC32__)
sl@0
   367
	_setmode(_fileno(inputFile), _O_BINARY);
sl@0
   368
	_setmode(_fileno(outputFile), _O_BINARY);
sl@0
   369
#endif
sl@0
   370
	HandleByteOrderMarks(outputByteOrderMark, unicodeByteOrder, inputEncoding, outputEncoding, inputFile, outputFile);
sl@0
   371
	DoConversion(unicodeByteOrder, inputEncoding, outputEncoding, inputFile, outputFile);
sl@0
   372
	FlushAndCloseFiles(inputFile, outputFile);
sl@0
   373
	return 0;
sl@0
   374
	}
sl@0
   375
sl@0
   376