/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2023 KiCad Developers, see AUTHORS.txt for contributors. * * This program is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation, either version 3 of the License, or (at your * option) any later version. * * 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. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program. If not, see . */ #include #include #include #include // Define USE_MSYS2_FALlBACK if the code for _MSC_VER does not compile on msys2 //#define USE_MSYS2_FALLBACK FILE* KIPLATFORM::IO::SeqFOpen( const wxString& aPath, const wxString& aMode ) { #if defined( _MSC_VER ) || !defined( USE_MSYS2_FALLBACK ) // We need to use the win32 api to setup a file handle with sequential scan flagged // and pass it up the chain to create a normal FILE stream HANDLE hFile = INVALID_HANDLE_VALUE; hFile = CreateFileW( aPath.wc_str(), GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL ); if (hFile == INVALID_HANDLE_VALUE) { return NULL; } int fd = _open_osfhandle( reinterpret_cast( hFile ), 0 ); if( fd == -1 ) { // close the handle manually as the ownership didnt transfer CloseHandle( hFile ); return NULL; } FILE* fp = _fdopen( fd, aMode.c_str() ); if( !fp ) { // close the file descriptor manually as the ownership didnt transfer _close( fd ); } return fp; #else // Fallback for MSYS2 return wxFopen( aPath, aMode ); #endif } bool KIPLATFORM::IO::DuplicatePermissions( const wxString &aSrc, const wxString &aDest ) { bool retval = false; DWORD dwSize = 0; // Retrieve the security descriptor from the source file if( GetFileSecurity( aSrc.wc_str(), OWNER_SECURITY_INFORMATION | GROUP_SECURITY_INFORMATION | DACL_SECURITY_INFORMATION, NULL, 0, &dwSize ) ) { #ifdef __MINGW32__ // pSD is used as PSECURITY_DESCRIPTOR, aka void* pointer // it create an annoying warning on gcc with "delete[] pSD;" : // "warning: deleting 'PSECURITY_DESCRIPTOR' {aka 'void*'} is undefined" // so use a BYTE* pointer (do not cast it to a void pointer) BYTE* pSD = new BYTE[dwSize]; #else PSECURITY_DESCRIPTOR pSD = static_cast( new BYTE[dwSize] ); #endif if( !pSD ) return false; if( !GetFileSecurity( aSrc.wc_str(), OWNER_SECURITY_INFORMATION | GROUP_SECURITY_INFORMATION | DACL_SECURITY_INFORMATION, pSD, dwSize, &dwSize ) ) { delete[] pSD; return false; } // Assign the retrieved security descriptor to the destination file if( !SetFileSecurity( aDest.wc_str(), OWNER_SECURITY_INFORMATION | GROUP_SECURITY_INFORMATION | DACL_SECURITY_INFORMATION, pSD ) ) { retval = false; } delete[] pSD; } return retval; } bool KIPLATFORM::IO::IsFileHidden( const wxString& aFileName ) { bool result = false; if( ( GetFileAttributesW( aFileName.fn_str() ) & FILE_ATTRIBUTE_HIDDEN ) ) result = true; return result; }