如何检测Windows下的FreePascal中是否重定向stdin或stdout?

时间:2016-11-20 20:45:49

标签: freepascal io-redirection

我需要知道是否已使用最新的FPC(v3.0.0)为控制台应用程序重定向stdin或stdout

在旧的Turbo Pascal时代,我有一个用汇编编写的函数来执行此操作。见下文:

{ **************************************************************
  * Routine   : RedirectedStd                                  *
  * Purpose   : Return Yes (True) if standard handle is being  *
  *           : redirected.                                    *
  * Note(s)   : Even though handle can take any handle value,  *
  *           : the function will be meaningful only for the   *
  *           : standard input, standard output, and standard  *
  *           : error.  It will, however, be True for any      *
  *           : handle that does NOT point to the console.     *
  *           : (Besides, this is what it actually checks for.)*
  *           : Make sure handle belongs to an open file or    *
  *           : you will get wrong answer (check IOResult).    *
  ************************************************************** }
function RedirectedStd(handle: Word): Boolean; assembler;
const
  DEVICE       = $0080;
  FASTCONSOLE  = $0010;
  CONSOUT      = $0002;
  CONSIN       = $0001;
asm
  mov     InOutRes,0
  mov     ax,$4400       { IOCTL svc, get device information }
  mov     bx,handle
  int     $21            { result in DX }
  mov     ax,1           { assume function is True }
  jc      @Error         { got error with code in AX }
  test    dx,DEVICE
  jz      @Out
  test    dx,FASTCONSOLE
  jz      @Out
  test    dx,CONSOUT
  jz      @Out
  test    dx,CONSIN
  jz      @Out
  xor     ax,ax          { function is False }
  jmp     @Out
@Error:
  mov     InOutRes,ax
@Out:
end; { RedirectedStd }

此语法对FPC汇编程序无效。我尝试了以下变体,虽然编译好但它崩溃了:

function RedirectedStd(handle: Word): Boolean; assembler;
label Error,Done;
const DEVICE       = $0080;
      FASTCONSOLE  = $0010;
      CONSOUT      = $0002;
      CONSIN       = $0001;
asm
          movw      $0,InOutRes
          movw      $4400,%ax           { IOCTL svc, get device information }
          movw      handle,%bx
          int       $21                 { result in DX }
          movw      $1,%ax              { assume function is True }
          jc        Error               { got error with code in AX }
          test      DEVICE,%dx
          jz        Done
          test      FASTCONSOLE,%dx
          jz        Done
          test      CONSOUT,%dx
          jz        Done
          test      CONSIN,%dx
          jz        Done
          xor       %ax,%ax             { function is False }
          jmp       Done
Error:    movw      %ax,InOutRes
Done:
end; { RedirectedStd }

(不确定我的转换是否相同。)

有什么想法吗?

编辑: 根据已接受的答案给了我足够的方向来找出解决方案,我想出了以下替代我原来的例程:

function RedirectedStd(handle: Word): Boolean; {$ifndef WINDOWS} unimplemented; {$endif}
begin
  RedirectedStd := False;
  {$ifdef WINDOWS}
  case handle of
    0: RedirectedStd := GetFileType(GetStdHandle(STD_INPUT_HANDLE)) <> FILE_TYPE_CHAR;
    1: RedirectedStd := GetFileType(GetStdHandle(STD_OUTPUT_HANDLE)) <> FILE_TYPE_CHAR;
    2: RedirectedStd := GetFileType(GetStdHandle(STD_ERROR_HANDLE)) <> FILE_TYPE_CHAR;
  end;
  {$endif}
end; { RedirectedStd }

哦,你需要使用Windows;

1 个答案:

答案 0 :(得分:1)

使用

GetFileType(GetStdHandle(STD_OUTPUT_HANDLE));

用于标准输出。应该明白该怎么做stdin。