2022-12-13 10:53:12 +00:00
|
|
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
|
|
// See the LICENCE file in the repository root for full licence text.
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using System.Runtime.InteropServices.ComTypes;
|
|
|
|
using Microsoft.Win32.SafeHandles;
|
2022-12-13 11:02:57 +00:00
|
|
|
using osu.Framework;
|
2022-12-13 10:53:12 +00:00
|
|
|
|
|
|
|
namespace osu.Game.IO
|
|
|
|
{
|
|
|
|
internal static class HardLinkHelper
|
|
|
|
{
|
2022-12-13 11:02:57 +00:00
|
|
|
public static bool CheckAvailability(string testDestinationPath, string testSourcePath)
|
|
|
|
{
|
2022-12-28 12:23:07 +00:00
|
|
|
// TODO: Add macOS support for hardlinks.
|
|
|
|
if (RuntimeInfo.OS != RuntimeInfo.Platform.Windows || RuntimeInfo.OS != RuntimeInfo.Platform.Linux)
|
2022-12-13 11:02:57 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
const string test_filename = "_hard_link_test";
|
|
|
|
|
|
|
|
testDestinationPath = Path.Combine(testDestinationPath, test_filename);
|
|
|
|
testSourcePath = Path.Combine(testSourcePath, test_filename);
|
|
|
|
|
|
|
|
cleanupFiles();
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
File.WriteAllText(testSourcePath, string.Empty);
|
|
|
|
// Test availability by creating an arbitrary hard link between the source and destination paths.
|
2022-12-28 12:23:07 +00:00
|
|
|
|
|
|
|
bool isHardLinkAvailable = false;
|
|
|
|
|
|
|
|
if (RuntimeInfo.OS == RuntimeInfo.Platform.Windows)
|
|
|
|
isHardLinkAvailable = CreateHardLink(testDestinationPath, testSourcePath, IntPtr.Zero);
|
|
|
|
else if (RuntimeInfo.OS == RuntimeInfo.Platform.Linux)
|
|
|
|
isHardLinkAvailable = link(testSourcePath, testDestinationPath) == 0;
|
|
|
|
|
|
|
|
return isHardLinkAvailable;
|
2022-12-13 11:02:57 +00:00
|
|
|
}
|
|
|
|
catch
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
finally
|
|
|
|
{
|
|
|
|
cleanupFiles();
|
|
|
|
}
|
|
|
|
|
|
|
|
void cleanupFiles()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
File.Delete(testDestinationPath);
|
|
|
|
File.Delete(testSourcePath);
|
|
|
|
}
|
|
|
|
catch
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-13 10:53:12 +00:00
|
|
|
// For future use (to detect if a file is a hard link with other references existing on disk).
|
|
|
|
public static int GetFileLinkCount(string filePath)
|
|
|
|
{
|
|
|
|
int result = 0;
|
|
|
|
SafeFileHandle handle = CreateFile(filePath, FileAccess.Read, FileShare.Read, IntPtr.Zero, FileMode.Open, FileAttributes.Archive, IntPtr.Zero);
|
|
|
|
|
|
|
|
ByHandleFileInformation fileInfo;
|
|
|
|
|
|
|
|
if (GetFileInformationByHandle(handle, out fileInfo))
|
|
|
|
result = (int)fileInfo.NumberOfLinks;
|
|
|
|
CloseHandle(handle);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-12-28 13:06:50 +00:00
|
|
|
[DllImport("libc", SetLastError = true)]
|
2022-12-28 12:23:07 +00:00
|
|
|
public static extern int link(string oldpath, string newpath);
|
|
|
|
|
2022-12-14 02:34:06 +00:00
|
|
|
[DllImport("Kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
|
2022-12-13 11:02:57 +00:00
|
|
|
public static extern bool CreateHardLink(string lpFileName, string lpExistingFileName, IntPtr lpSecurityAttributes);
|
2022-12-13 10:53:12 +00:00
|
|
|
|
|
|
|
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Auto)]
|
|
|
|
private static extern SafeFileHandle CreateFile(
|
|
|
|
string lpFileName,
|
|
|
|
[MarshalAs(UnmanagedType.U4)] FileAccess dwDesiredAccess,
|
|
|
|
[MarshalAs(UnmanagedType.U4)] FileShare dwShareMode,
|
|
|
|
IntPtr lpSecurityAttributes,
|
|
|
|
[MarshalAs(UnmanagedType.U4)] FileMode dwCreationDisposition,
|
|
|
|
[MarshalAs(UnmanagedType.U4)] FileAttributes dwFlagsAndAttributes,
|
|
|
|
IntPtr hTemplateFile);
|
|
|
|
|
|
|
|
[DllImport("kernel32.dll", SetLastError = true)]
|
|
|
|
private static extern bool GetFileInformationByHandle(SafeFileHandle handle, out ByHandleFileInformation lpFileInformation);
|
|
|
|
|
|
|
|
[DllImport("kernel32.dll", SetLastError = true)]
|
|
|
|
[return: MarshalAs(UnmanagedType.Bool)]
|
|
|
|
private static extern bool CloseHandle(SafeHandle hObject);
|
2022-12-13 11:02:57 +00:00
|
|
|
|
|
|
|
[StructLayout(LayoutKind.Sequential)]
|
|
|
|
private struct ByHandleFileInformation
|
|
|
|
{
|
|
|
|
public readonly uint FileAttributes;
|
|
|
|
public readonly FILETIME CreationTime;
|
|
|
|
public readonly FILETIME LastAccessTime;
|
|
|
|
public readonly FILETIME LastWriteTime;
|
|
|
|
public readonly uint VolumeSerialNumber;
|
|
|
|
public readonly uint FileSizeHigh;
|
|
|
|
public readonly uint FileSizeLow;
|
|
|
|
public readonly uint NumberOfLinks;
|
|
|
|
public readonly uint FileIndexHigh;
|
|
|
|
public readonly uint FileIndexLow;
|
|
|
|
}
|
2022-12-13 10:53:12 +00:00
|
|
|
}
|
|
|
|
}
|