fpc/compiler/widestr.pas
Jonas Maebe 05bf826342 + support UTF-8 in ascii2unicode(), this fixes the UTF-16 output of
resourcestring data .rsj files in case the source file is interpreted as
    UTF-8. Previously, the individual UTF-8 bytes were each stored in a
    separate widechar in the Json file (mantis #28717)

  * due to the fact that rstconv didn't use the cwstring unit on Unix, rstconv
    until now just concatenated the bytes stored in the widechars of the Json
    file on those platforms, i.e., the strings put in the resource file were
    byte for byte equal to what was in the source file. On Windows, these bytes
    were interpreted as individual widechars, converted to the
    DefaultSystemCodePage and then written. This means that for anything but
    ISO-8859-1 (where every widechar from #0000 to #0255 maps to #0 to #255),
    the output got corrupted.

    In order to keep compatibility with the old behaviour whereby rstconv wrote
    the resource strings using the same encoding as in the source file (except
    if the data got completely corrupted, in which case compatibility is
    useless), we now store all resourcestrings twice in the .rsj file: once as
    the exact byte sequence from the source file, and once (properly) encoded
    in UTF-16.

    By default, rstconv will use the byte string and just write that one to the
    resource file. Additionally, there is a new -p option that accepts a code
    page name (see rstconv -h for the list of supported names), which can be
    used to make rstconv use the UTF-16 version and convert that to the desired
    code page (as long as the system on which rstconv runs supports that
    codepage).

    And this also finally resolves mantis #6477.

git-svn-id: trunk@31881 -
2015-09-28 22:14:56 +00:00

410 lines
12 KiB
ObjectPascal

{
Copyright (c) 2000-2002 by Florian Klaempfl
This unit contains basic functions for unicode support in the
compiler, this unit is mainly necessary to bootstrap widestring
support ...
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
****************************************************************************
}
unit widestr;
{$i fpcdefs.inc}
interface
uses
{$if FPC_FULLVERSION<20700}ccharset{$else}charset{$endif},globtype;
type
tcompilerwidechar = word;
tcompilerwidecharptr = ^tcompilerwidechar;
pcompilerwidechar = ^tcompilerwidechar;
pcompilerwidestring = ^_tcompilerwidestring;
_tcompilerwidestring = record
data : pcompilerwidechar;
maxlen,len : SizeInt;
end;
procedure initwidestring(out r : pcompilerwidestring);
procedure donewidestring(var r : pcompilerwidestring);
procedure setlengthwidestring(r : pcompilerwidestring;l : SizeInt);
function getlengthwidestring(r : pcompilerwidestring) : SizeInt;
procedure concatwidestringchar(r : pcompilerwidestring;c : tcompilerwidechar);
procedure concatwidestrings(s1,s2 : pcompilerwidestring);
function comparewidestrings(s1,s2 : pcompilerwidestring) : SizeInt;
procedure copywidestring(s,d : pcompilerwidestring);
function asciichar2unicode(c : char) : tcompilerwidechar;
function unicode2asciichar(c : tcompilerwidechar) : char;
procedure ascii2unicode(p : pchar;l : SizeInt;cp : tstringencoding;r : pcompilerwidestring;codepagetranslation : boolean = true);
procedure unicode2ascii(r : pcompilerwidestring;p : pchar;cp : tstringencoding);
function hasnonasciichars(const p: pcompilerwidestring): boolean;
function getcharwidestring(r : pcompilerwidestring;l : SizeInt) : tcompilerwidechar;
function cpavailable(const s: string) : boolean;
function cpavailable(cp: word) : boolean;
procedure changecodepage(
s : pchar; l : SizeInt; scp : tstringencoding;
d : pchar; dcp : tstringencoding
);
function codepagebyname(const s : string) : tstringencoding;
function charlength(p: pchar; len: sizeint): sizeint;
function charlength(const s: string): sizeint;
implementation
uses
{$if FPC_FULLVERSION>20700}
{ use only small codepage maps, others will be }
{ loaded on demand from -FM path }
{ cyrillic code pages }
cp1251,cp866,cp8859_5,
{ greek code page }
cp1253,
{ other code pages }
cp8859_1,cp850,cp437,cp1252,cp646,
cp874, cp856,cp852,cp8859_2,
cp1250,cp1254,cp1255,cp1256,cp1257,cp1258,
{$endif}
globals,cutils;
procedure initwidestring(out r : pcompilerwidestring);
begin
new(r);
r^.data:=nil;
r^.len:=0;
r^.maxlen:=0;
end;
procedure donewidestring(var r : pcompilerwidestring);
begin
if assigned(r^.data) then
freemem(r^.data);
dispose(r);
r:=nil;
end;
function getcharwidestring(r : pcompilerwidestring;l : SizeInt) : tcompilerwidechar;
begin
getcharwidestring:=r^.data[l];
end;
function getlengthwidestring(r : pcompilerwidestring) : SizeInt;
begin
getlengthwidestring:=r^.len;
end;
procedure growwidestring(r : pcompilerwidestring;l : SizeInt);
begin
if r^.maxlen>=l then
exit;
if assigned(r^.data) then
reallocmem(r^.data,sizeof(tcompilerwidechar)*l)
else
getmem(r^.data,sizeof(tcompilerwidechar)*l);
r^.maxlen:=l;
end;
procedure setlengthwidestring(r : pcompilerwidestring;l : SizeInt);
begin
r^.len:=l;
if l>r^.maxlen then
growwidestring(r,l);
end;
procedure concatwidestringchar(r : pcompilerwidestring;c : tcompilerwidechar);
begin
if r^.len>=r^.maxlen then
growwidestring(r,r^.len+16);
r^.data[r^.len]:=c;
inc(r^.len);
end;
procedure concatwidestrings(s1,s2 : pcompilerwidestring);
begin
growwidestring(s1,s1^.len+s2^.len);
move(s2^.data^,s1^.data[s1^.len],s2^.len*sizeof(tcompilerwidechar));
inc(s1^.len,s2^.len);
end;
procedure copywidestring(s,d : pcompilerwidestring);
begin
setlengthwidestring(d,s^.len);
move(s^.data^,d^.data^,s^.len*sizeof(tcompilerwidechar));
end;
function comparewidestrings(s1,s2 : pcompilerwidestring) : SizeInt;
var
maxi,temp : SizeInt;
begin
if pointer(s1)=pointer(s2) then
begin
comparewidestrings:=0;
exit;
end;
maxi:=s1^.len;
temp:=s2^.len;
if maxi>temp then
maxi:=Temp;
temp:=compareword(s1^.data^,s2^.data^,maxi);
if temp=0 then
temp:=s1^.len-s2^.len;
comparewidestrings:=temp;
end;
function asciichar2unicode(c : char) : tcompilerwidechar;
var
m : punicodemap;
begin
if (current_settings.sourcecodepage <> CP_UTF8) then
begin
m:=getmap(current_settings.sourcecodepage);
asciichar2unicode:=getunicode(c,m);
end
else
result:=tcompilerwidechar(c);
end;
function unicode2asciichar(c : tcompilerwidechar) : char;
{begin
if word(c)<128 then
unicode2asciichar:=char(word(c))
else
unicode2asciichar:='?';
end;}
begin
Result := getascii(c,getmap(current_settings.sourcecodepage))[1];
end;
procedure ascii2unicode(p : pchar;l : SizeInt;cp : tstringencoding;r : pcompilerwidestring;codepagetranslation : boolean = true);
var
source : pchar;
dest : tcompilerwidecharptr;
i : SizeInt;
m : punicodemap;
begin
m:=getmap(cp);
setlengthwidestring(r,l);
source:=p;
dest:=tcompilerwidecharptr(r^.data);
if codepagetranslation then
begin
if cp<>CP_UTF8 then
begin
for i:=1 to l do
begin
dest^:=getunicode(source^,m);
inc(dest);
inc(source);
end;
end
else
begin
r^.len:=Utf8ToUnicode(punicodechar(r^.data),r^.maxlen,p,l);
{ -1, because utf8tounicode includes room for a terminating 0 in
its result count }
if r^.len>0 then
dec(r^.len);
end;
end
else
begin
for i:=1 to l do
begin
dest^:=tcompilerwidechar(source^);
inc(dest);
inc(source);
end;
end;
end;
procedure unicode2ascii(r : pcompilerwidestring;p:pchar;cp : tstringencoding);
var
m : punicodemap;
source : tcompilerwidecharptr;
dest : pchar;
i : longint;
begin
if (cp = 0) or (cp=CP_NONE) then
m:=getmap(current_settings.sourcecodepage)
else
m:=getmap(cp);
// !!!! MBCS
source:=tcompilerwidecharptr(r^.data);
dest:=p;
for i:=1 to r^.len do
begin
dest^ := getascii(source^,m)[1];
inc(dest);
inc(source);
end;
end;
(*
var
source : tcompilerwidecharptr;
dest : pchar;
i : longint;
begin
{ This routine must work the same as the
the routine in the RTL to have the same compile time (for constant strings)
and runtime conversion (for variables) }
source:=tcompilerwidecharptr(r^.data);
dest:=p;
for i:=1 to r^.len do
begin
if word(source^)<128 then
dest^:=char(word(source^))
else
dest^:='?';
inc(dest);
inc(source);
end;
end;
*)
function hasnonasciichars(const p: pcompilerwidestring): boolean;
var
source : tcompilerwidecharptr;
i : longint;
begin
source:=tcompilerwidecharptr(p^.data);
result:=true;
for i:=1 to p^.len do
begin
if word(source^)>=128 then
exit;
inc(source);
end;
result:=false;
end;
function cpavailable(const s: string): boolean;
begin
result:=mappingavailable(lower(s));
{$if FPC_FULLVERSION>20700}
if not result then
result:=(unicodepath<>'')and(registerbinarymapping(unicodepath+'charset',lower(s)));
{$ifend}
end;
function cpavailable(cp: word): boolean;
begin
result:=mappingavailable(cp);
{$if FPC_FULLVERSION>20700}
if not result then
result:=(unicodepath<>'')and(registerbinarymapping(unicodepath+'charset','cp'+tostr(cp)));
{$ifend}
end;
procedure changecodepage(
s : pchar; l : SizeInt; scp : tstringencoding;
d : pchar; dcp : tstringencoding
);
var
ms, md : punicodemap;
source : pchar;
dest : pchar;
i : longint;
begin
ms:=getmap(scp);
md:=getmap(dcp);
source:=s;
dest:=d;
for i:=1 to l do
begin
dest^ := getascii(getunicode(source^,ms),md)[1];
inc(dest);
inc(source);
end;
end;
function codepagebyname(const s : string) : tstringencoding;
var
p : punicodemap;
begin
Result:=0;
p:=getmap(s);
if (p<>nil) then
Result:=p^.cp;
end;
function charlength(p: pchar; len: sizeint): sizeint;
{$IFDEF FPC_HAS_CPSTRING}
var
p2: pchar;
i, chars, codepointlen: sizeint;
{$ENDIF FPC_HAS_CPSTRING}
begin
{$IFDEF FPC_HAS_CPSTRING}
if len=0 then
begin
result:=0;
exit;
end;
{ Length of the string converted to a SBCS codepage (e.g. ISO 8859-1)
should be equal to the amount of characters in the source string. }
if defaultsystemcodepage=cp_utf8 then
{ ChangeCodePage does not work for UTF-8 apparently... :-( }
begin
i:=1;
chars:=0;
while i<=len do
begin
codepointlen:=utf8codepointlen(p,len-i+1,true);
inc(i,codepointlen);
inc(p,codepointlen);
inc(chars);
end;
result:=chars;
end
else if cpavailable(defaultsystemcodepage) then
begin
getmem(p2,succ(len));
fillchar(p2^,succ(len),0);
changecodepage(p,len,defaultsystemcodepage,p2,28591);
result:=strlen(p2);
freemem(p2,succ(len));
end
else
result:=len;
{$ELSE FPC_HAS_CPSTRING}
result:=len;
{$ENDIF FPC_HAS_CPSTRING}
end;
function charlength(const s: string): sizeint;
begin
result:=charlength(@s[1],length(s));
end;
end.