fpc/rtl/nativent/ndkutils.pas
2023-07-27 19:04:03 +02:00

177 lines
4.7 KiB
ObjectPascal

{
FPC Utility Functions for Native NT applications
This file is part of the Free Pascal run time library.
Copyright (c) 2009 by Sven Barth
See the file COPYING.FPC, included in this distribution,
for details about the copyright.
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.
**********************************************************************}
{$IFNDEF FPC_DOTTEDUNITS}
unit NDKUtils;
{$ENDIF FPC_DOTTEDUNITS}
{$mode objfpc}{$H+}
interface
{$IFDEF FPC_DOTTEDUNITS}
uses
NTApi.NDK;
{$ELSE FPC_DOTTEDUNITS}
uses
NDK;
{$ENDIF FPC_DOTTEDUNITS}
// Helpers for converting Pascal string types to NT's UNICODE_STRING
procedure ShortStrToNTStr(aStr: ShortString; var aNTStr: UNICODE_STRING);
procedure AnsiStrToNTStr(const aStr: RawByteString; var aNTStr: UNICODE_STRING);
procedure UnicodeStrToNtStr(const aStr: UnicodeString;
var aNTStr: UNICODE_STRING);
procedure PCharToNTStr(aStr: PAnsiChar; aLen: Cardinal; var aNTStr: UNICODE_STRING);
procedure FreeNTStr(var aNTStr: UNICODE_STRING);
// Wraps NtDisplayString for use with Write(Ln)
procedure AssignDisplayString(var aFile: Text; aUtf8: Boolean);
implementation
{$IFDEF FPC_DOTTEDUNITS}
uses
System.SysUtils;
{$ELSE FPC_DOTTEDUNITS}
uses
SysUtils;
{$ENDIF FPC_DOTTEDUNITS}
procedure ShortStrToNTStr(aStr: ShortString; var aNTStr: UNICODE_STRING);
var
buf: Pointer;
i: Integer;
begin
aNTStr.Length := Length(aStr) * 2;
aNTStr.buffer := GetMem(aNTStr.Length);
buf := aNTStr.buffer;
for i := 1 to Length(aStr) do begin
PWord(buf)^ := Word(aStr[i]);
buf := Pointer(PtrUInt(buf) + SizeOf(Word));
end;
aNTStr.MaximumLength := aNTStr.Length;
end;
procedure AnsiStrToNTStr(const aStr: RawByteString; var aNTStr: UNICODE_STRING);
var
buf: PWideChar;
i: Integer;
begin
aNTStr.Length := Length(aStr) * 2;
aNTStr.Buffer := GetMem(aNTStr.Length);
buf := aNTStr.buffer;
for i := 1 to Length(aStr) do begin
buf^ := WideChar(Word(aStr[i]));
Inc(buf);
end;
aNTStr.MaximumLength := aNTStr.Length;
end;
procedure UnicodeStrToNtStr(const aStr: UnicodeString;
var aNTStr: UNICODE_STRING);
var
buf: PWideChar;
begin
{ TODO : check why this prints garbage }
aNTStr.Length := Length(aStr) * 2;
aNTStr.Buffer := GetMem(aNTStr.Length);
if Length(aStr) > 0 then
Move(aStr[1], aNTStr.Buffer^, aNTStr.Length);
aNTStr.MaximumLength := aNTStr.Length;
end;
procedure PCharToNTStr(aStr: PAnsiChar; aLen: Cardinal; var aNTStr: UNICODE_STRING);
var
i: Integer;
begin
if (aLen = 0) and (aStr <> Nil) and (aStr^ <> #0) then
aLen := StrLen(aStr);
aNtStr.Length := aLen * SizeOf(WideChar);
aNtStr.MaximumLength := aNtStr.Length;
aNtStr.Buffer := GetMem(aNtStr.Length);
for i := 0 to aLen do
aNtStr.Buffer[i] := aStr[i];
end;
procedure FreeNTStr(var aNTStr: UNICODE_STRING);
begin
if aNTStr.Buffer <> Nil then
FreeMem(aNTStr.Buffer);
FillChar(aNTStr, SizeOf(UNICODE_STRING), 0);
end;
function DisplayStringWriteFunc(var aFile: TTextRec ): LongInt;
var
ntstr: TNtUnicodeString;
len: SizeUInt;
begin
Result := 0;
with aFile do
if (BufPos>0) then begin
if Boolean(UserData[1]) then begin
{ TODO : check why UTF8 prints garbage }
{len := Utf8ToUnicode(Nil, 0, PAnsiChar(BufPtr), BufPos);
ntstr.Length := len * 2;
ntstr.MaximumLength := ntstr.Length;
ntstr.Buffer := GetMem(ntstr.Length);
Utf8ToUnicode(ntstr.Buffer, len, PAnsiChar(BufPtr), BufPos);}
PCharToNtStr(PAnsiChar(BufPtr), BufPos, ntstr);
end else
PCharToNtStr(PAnsiChar(BufPtr), BufPos, ntstr);
NtDisplayString(@ntstr);
// FreeNTStr uses FreeMem, so we don't need an If here
FreeNtStr(ntstr);
BufPos := 0;
end;
end;
function DisplayStringCloseFunc(var aFile: TTextRec): LongInt;
begin
Result := 0;
end;
function DisplayStringOpenFunc(var aFile: TTextRec ): LongInt;
begin
Result := 0;
end;
procedure AssignDisplayString(var aFile: Text; aUtf8: Boolean);
begin
FillChar(aFile, SizeOf(TextRec), 0);
{ only set things that are not zero }
TextRec(aFile).Handle := UnusedHandle;
TextRec(aFile).mode := fmOutput;
TextRec(aFile).BufSize := TextRecBufSize;
TextRec(aFile).Bufptr := @TextRec(aFile).Buffer;
TextRec(aFile).OpenFunc := @DisplayStringOpenFunc;
case DefaultTextLineBreakStyle of
tlbsLF:
TextRec(aFile).LineEnd := #10;
tlbsCRLF:
TextRec(aFile).LineEnd := #13#10;
tlbsCR:
TextRec(aFile).LineEnd := #13;
end;
TextRec(aFile).Closefunc := @DisplayStringCloseFunc;
TextRec(aFile).InOutFunc := @DisplayStringWriteFunc;
TextRec(aFile).FlushFunc := @DisplayStringWriteFunc;
TextRec(aFile).UserData[1] := Ord(aUTF8);
end;
end.