[fpc-devel] Standard in/out/error of Windows GUI apps

Henry Vermaak henry.vermaak at gmail.com
Sun Feb 12 13:54:58 CET 2017


Hello,

Recently a customer asked whether he could get a GUI version of one of
our console apps because the flashing console is annoying him.  Fair
enough, I thought, this should be easy.  We do it with our C utilities
already.

I know that a writeln() in a GUI app crashes, from when someone left a
stray writeln() in FPC 3.0.x and our app started crashing when looking
for updates online.  Can someone explain to me what the use of this is?
If you know that a handle doesn't exist, why not just redirect the
output to 'nul'?

I set out to do this myself, when I discovered that fpc tries to show
anything that you write to stdout/stderr in a messagebox.  It does this
even when the handles are being redirected (to e.g. pipes) by the
calling application.  Relevant code in syswin.inc (procedure
SysInitStdIO):

if not IsConsole then
 begin
  AssignError(stderr);
  AssignError(StdOut);
  Assign(Output,'');
  Assign(Input,'');
  Assign(ErrOutput,'');
 end
...
...

Showing writeln output in a messagebox is questionable, but it doesn't
even do it for Output and ErrOutput.  The documentation says they are
aliases for stdout/stderr?  So not only does it ignore whichever
possibly valid handles in Std{Output|Input|Error}Handle, but it also
leaves ErrOutput pointing to StdOutputHandle.

For anyone interested, this is what I had to do so that a command like
"app > stdout.txt 2> stderr.txt" behaves as expected:

{$ifdef windows}
function CanOpenHandle(AHandle: THandle): Boolean;
begin
  Result := GetFileType(AHandle) in [FILE_TYPE_DISK, FILE_TYPE_PIPE, FILE_TYPE_CHAR];
end;

procedure RedirectStdIO(var AFile: Text; AHandle: THandle);
var
  CanOpen: Boolean;
begin
  CanOpen := CanOpenHandle(AHandle);
  AssignFile(AFile, IfThen(CanOpen, '', 'nul'));

  if AHandle = StdOutputHandle then begin
    Rewrite(AFile);
  end else if AHandle = StdErrorHandle then begin
    Rewrite(AFile);
    { Rewriting a file with a '' name will use StdOutputHandle, but we want
      StdErrorHandle, so hack it in. }
    if CanOpen then
      TextRec(AFile).Handle := AHandle;
  end else if AHandle = StdInputHandle then begin
    Reset(AFile);
  end;
end;

procedure RedirectWinGUIStdIO;
begin
  if IsConsole then
    Exit;

  RedirectStdIO(StdOut, StdOutputHandle);
  RedirectStdIO(Output, StdOutputHandle);

  RedirectStdIO(StdErr, StdErrorHandle);
  RedirectStdIO(ErrOutput, StdErrorHandle);

  RedirectStdIO(Input, StdInputHandle);
end;
{$endif}

Then call RedirectWinGUIStdIO at the start.  I hope it works for all
possibilities, my Windows skills are lacking.

Henry



More information about the fpc-devel mailing list