From 46b98a6da49f141e4e05b47bc86e881d4dca5bad Mon Sep 17 00:00:00 2001 From: nickysn Date: Tue, 2 Apr 2019 17:44:23 +0000 Subject: [PATCH] * Improvements to the implementation of FpOpenDir for *BSD: * use FpFStat instead of FpStat to avoid a race condition, where the directory path could change between the calls of FpOpen and FpStat * improved cleanup (i.e. call FpClose, free allocated memory to avoid memleaks) in case of errors git-svn-id: trunk@41820 - --- rtl/bsd/ossysc.inc | 33 ++++++++++++++++++++++++--------- 1 file changed, 24 insertions(+), 9 deletions(-) diff --git a/rtl/bsd/ossysc.inc b/rtl/bsd/ossysc.inc index df0317260a..ef788cde77 100644 --- a/rtl/bsd/ossysc.inc +++ b/rtl/bsd/ossysc.inc @@ -172,22 +172,17 @@ end; const DIRBLKSIZ=1024; +function Fpfstat(fd : cint; var sb : stat): cint; forward; + function Fpopendir(dirname : pchar): pdir; [public, alias : 'FPC_SYSC_OPENDIR']; var fd:longint; st:stat; ptr:pdir; + save_errno:cint; begin Fpopendir:=nil; - if Fpstat(dirname,st)<0 then - exit; -{ Is it a dir ? } - if not((st.st_mode and $f000)=$4000)then - begin - errno:=ESysENOTDIR; - exit - end; { Open it} fd:=Fpopen(dirname,O_RDONLY,438); if fd<0 then @@ -195,15 +190,35 @@ begin Errno:=-1; exit; End; + if FpFStat(fd,st)<0 then + begin + save_errno:=errno; + FpClose(fd); + errno:=save_errno; + exit; + end; +{ Is it a dir ? } + if not((st.st_mode and $f000)=$4000)then + begin + FpClose(fd); + errno:=ESysENOTDIR; + exit + end; new(ptr); if ptr=nil then Begin + FpClose(fd); Errno:=1; exit; End; Getmem(ptr^.dd_buf,2*DIRBLKSIZ); if ptr^.dd_buf=nil then - exit; + begin + dispose(ptr); + FpClose(fd); + Errno:=1; + exit; + end; ptr^.dd_fd:=fd; ptr^.dd_loc:=-1; ptr^.dd_rewind:=ptrint(ptr^.dd_buf);