|
@@ -0,0 +1,188 @@
|
|
|
+{
|
|
|
+ Double Commander
|
|
|
+ -------------------------------------------------------------------------
|
|
|
+ This unit contains Haiku specific functions
|
|
|
+
|
|
|
+ Copyright (C) 2022 Alexander Koblov ([email protected])
|
|
|
+
|
|
|
+ Permission is hereby granted, free of charge, to any person obtaining
|
|
|
+ a copy of this software and associated documentation files (the
|
|
|
+ "Software"), to deal in the Software without restriction, including
|
|
|
+ without limitation the rights to use, copy, modify, merge, publish,
|
|
|
+ distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
+ permit persons to whom the Software is furnished to do so, subject to
|
|
|
+ the following conditions:
|
|
|
+
|
|
|
+ The above copyright notice and this permission notice shall be included
|
|
|
+ in all copies or substantial portions of the Software.
|
|
|
+
|
|
|
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
|
+ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
|
+ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
|
+ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
|
+ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
+}
|
|
|
+
|
|
|
+unit DCHaiku;
|
|
|
+
|
|
|
+{$mode objfpc}{$H+}
|
|
|
+{$packrecords c}
|
|
|
+
|
|
|
+interface
|
|
|
+
|
|
|
+uses
|
|
|
+ Classes, SysUtils, BaseUnix, Unix;
|
|
|
+
|
|
|
+const
|
|
|
+ O_NOTRAVERSE = $2000;
|
|
|
+
|
|
|
+type
|
|
|
+ attr_info = record
|
|
|
+ type_: cuint32;
|
|
|
+ size: coff_t;
|
|
|
+ end;
|
|
|
+ Tattr_info = attr_info;
|
|
|
+ Pattr_info = ^attr_info;
|
|
|
+
|
|
|
+function fs_remove_attr(fd: cint; attribute: PAnsiChar): cint; cdecl; external clib;
|
|
|
+function fs_stat_attr(fd: cint; attribute: pansichar; attrInfo: Pattr_info): cint; cdecl; external clib;
|
|
|
+
|
|
|
+function fs_open_attr(path: pansichar; attribute: pansichar;
|
|
|
+ type_: cuint32; openMode: cint): cint; cdecl; external clib;
|
|
|
+function fs_fopen_attr(fd: cint; attribute: pansichar;
|
|
|
+ type_: cuint32; openMode: cint): cint; cdecl; external clib;
|
|
|
+function fs_close_attr(fd: cint): cint; cdecl; external clib;
|
|
|
+
|
|
|
+function fs_open_attr_dir(path: pansichar): pDir; cdecl; external clib;
|
|
|
+function fs_lopen_attr_dir(path: pansichar): pDir; cdecl; external clib;
|
|
|
+function fs_fopen_attr_dir(fd: cint): pDir; cdecl; external clib;
|
|
|
+function fs_close_attr_dir(dir: pDir): cint; cdecl; external clib;
|
|
|
+function fs_read_attr_dir(dir: pDir): pDirent; cdecl; external clib;
|
|
|
+
|
|
|
+function mbFileCopyXattr(const Source, Target: String): Boolean;
|
|
|
+
|
|
|
+implementation
|
|
|
+
|
|
|
+uses
|
|
|
+ DCUnix;
|
|
|
+
|
|
|
+function mbFileOpen(const FileName: String; Flags: cInt): THandle;
|
|
|
+begin
|
|
|
+ repeat
|
|
|
+ Result:= fpOpen(FileName, Flags or O_CLOEXEC);
|
|
|
+ until (Result <> -1) or (fpgeterrno <> ESysEINTR);
|
|
|
+end;
|
|
|
+
|
|
|
+function mbFileGetXattr(hFile: THandle): TStringArray;
|
|
|
+var
|
|
|
+ DirPtr: pDir;
|
|
|
+ PtrDirEnt: pDirent;
|
|
|
+ Index: Integer = 0;
|
|
|
+begin
|
|
|
+ Result:= Default(TStringArray);
|
|
|
+ DirPtr:= fs_fopen_attr_dir(hFile);
|
|
|
+ if Assigned(DirPtr) then
|
|
|
+ try
|
|
|
+ SetLength(Result, 512);
|
|
|
+ PtrDirEnt:= fs_read_attr_dir(DirPtr);
|
|
|
+ while PtrDirEnt <> nil do
|
|
|
+ begin
|
|
|
+ Result[Index]:= StrPas(PtrDirEnt^.d_name);
|
|
|
+ Inc(Index);
|
|
|
+
|
|
|
+ if (Index > High(Result)) then
|
|
|
+ begin
|
|
|
+ SetLength(Result, Length(Result) * 2);
|
|
|
+ end;
|
|
|
+
|
|
|
+ PtrDirEnt:= fs_read_attr_dir(DirPtr);
|
|
|
+ end;
|
|
|
+ finally
|
|
|
+ fs_close_attr_dir(DirPtr);
|
|
|
+ end;
|
|
|
+ SetLength(Result, Index);
|
|
|
+end;
|
|
|
+
|
|
|
+function mbFileCopyXattr(const Source, Target: String): Boolean;
|
|
|
+var
|
|
|
+ AData: TBytes;
|
|
|
+ Index: Integer;
|
|
|
+ Names: TStringArray;
|
|
|
+ AttrName: PAnsiChar;
|
|
|
+ AttrInfo: Tattr_info;
|
|
|
+ hSource, hTarget: THandle;
|
|
|
+
|
|
|
+ function ReadAttr(hFile: cint; attribute: pansichar): Boolean;
|
|
|
+ var
|
|
|
+ hAttr: THandle;
|
|
|
+ begin
|
|
|
+ hAttr:= fs_fopen_attr(hFile, attribute, attrInfo.type_, O_RDONLY);
|
|
|
+ Result:= (hAttr <> feInvalidHandle);
|
|
|
+ if Result then
|
|
|
+ begin
|
|
|
+ Result:= (FileRead(hAttr, AData[0], attrInfo.size) = attrInfo.size);
|
|
|
+ fs_close_attr(hAttr);
|
|
|
+ end;
|
|
|
+ end;
|
|
|
+
|
|
|
+ function WriteAttr(hFile: cint; attribute: pansichar): Boolean;
|
|
|
+ var
|
|
|
+ hAttr: THandle;
|
|
|
+ begin
|
|
|
+ hAttr:= fs_fopen_attr(hFile, attribute, attrInfo.type_, O_CREAT or O_WRONLY);
|
|
|
+ Result:= (hAttr <> feInvalidHandle);
|
|
|
+ if Result then
|
|
|
+ begin
|
|
|
+ Result:= (FileWrite(hAttr, AData[0], attrInfo.size) = attrInfo.size);
|
|
|
+ fs_close_attr(hAttr);
|
|
|
+ end;
|
|
|
+ end;
|
|
|
+
|
|
|
+begin
|
|
|
+ hSource:= mbFileOpen(Source, O_RDONLY or O_NOTRAVERSE);
|
|
|
+ Result:= (hSource <> feInvalidHandle);
|
|
|
+ if Result then
|
|
|
+ begin
|
|
|
+ hTarget:= mbFileOpen(Target, O_RDWR or O_NOTRAVERSE);
|
|
|
+ Result:= (hTarget <> feInvalidHandle);
|
|
|
+ if Result then
|
|
|
+ begin
|
|
|
+ // Remove attributes from target
|
|
|
+ Names:= mbFileGetXattr(hTarget);
|
|
|
+ for Index:= 0 to High(Names) do
|
|
|
+ begin
|
|
|
+ fs_remove_attr(hTarget, PAnsiChar(Names[Index]));
|
|
|
+ end;
|
|
|
+
|
|
|
+ SetLength(AData, $FFFF);
|
|
|
+ Names:= mbFileGetXattr(hSource);
|
|
|
+
|
|
|
+ for Index:= 0 to High(Names) do
|
|
|
+ begin
|
|
|
+ AttrName:= PAnsiChar(Names[Index]);
|
|
|
+
|
|
|
+ if (fs_stat_attr(hSource, AttrName, @AttrInfo) >= 0) then
|
|
|
+ begin
|
|
|
+ if (AttrInfo.size > Length(AData)) then
|
|
|
+ begin
|
|
|
+ SetLength(AData, AttrInfo.size);
|
|
|
+ end;
|
|
|
+
|
|
|
+ Result:= ReadAttr(hSource, AttrName);
|
|
|
+ if Result then
|
|
|
+ begin
|
|
|
+ Result:= WriteAttr(hTarget, AttrName);
|
|
|
+ if not Result then Break;
|
|
|
+ end;
|
|
|
+ end;
|
|
|
+ end;
|
|
|
+ FileClose(hTarget);
|
|
|
+ end;
|
|
|
+ FileClose(hSource);
|
|
|
+ end;
|
|
|
+end;
|
|
|
+
|
|
|
+end.
|
|
|
+
|