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 |
|