mirror of https://github.com/ppy/osu
191 lines
6.7 KiB
C#
191 lines
6.7 KiB
C#
// 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;
|
|
using osu.Framework;
|
|
|
|
namespace osu.Game.IO
|
|
{
|
|
internal static class HardLinkHelper
|
|
{
|
|
public static bool CheckAvailability(string testDestinationPath, string testSourcePath)
|
|
{
|
|
// For simplicity, only support desktop operating systems for now.
|
|
if (!RuntimeInfo.IsDesktop)
|
|
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.
|
|
return TryCreateHardLink(testDestinationPath, testSourcePath);
|
|
}
|
|
catch
|
|
{
|
|
return false;
|
|
}
|
|
finally
|
|
{
|
|
cleanupFiles();
|
|
}
|
|
|
|
void cleanupFiles()
|
|
{
|
|
try
|
|
{
|
|
File.Delete(testDestinationPath);
|
|
File.Delete(testSourcePath);
|
|
}
|
|
catch
|
|
{
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempts to create a hard link from <paramref name="sourcePath"/> to <paramref name="destinationPath"/>,
|
|
/// using platform-specific native methods.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Hard links are only available on desktop platforms.
|
|
/// </remarks>
|
|
/// <returns>Whether the hard link was successfully created.</returns>
|
|
public static bool TryCreateHardLink(string destinationPath, string sourcePath)
|
|
{
|
|
switch (RuntimeInfo.OS)
|
|
{
|
|
case RuntimeInfo.Platform.Windows:
|
|
return CreateHardLink(destinationPath, sourcePath, IntPtr.Zero);
|
|
|
|
case RuntimeInfo.Platform.Linux:
|
|
case RuntimeInfo.Platform.macOS:
|
|
return link(sourcePath, destinationPath) == 0;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// 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;
|
|
|
|
switch (RuntimeInfo.OS)
|
|
{
|
|
case RuntimeInfo.Platform.Windows:
|
|
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);
|
|
break;
|
|
|
|
case RuntimeInfo.Platform.Linux:
|
|
case RuntimeInfo.Platform.macOS:
|
|
if (stat(filePath, out var statbuf) == 0)
|
|
result = (int)statbuf.st_nlink;
|
|
|
|
break;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
#region Windows native methods
|
|
|
|
[DllImport("Kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
|
|
public static extern bool CreateHardLink(string lpFileName, string lpExistingFileName, IntPtr lpSecurityAttributes);
|
|
|
|
[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);
|
|
|
|
[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;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Linux native methods
|
|
|
|
#pragma warning disable IDE1006 // Naming rule violation
|
|
|
|
[DllImport("libc", SetLastError = true)]
|
|
public static extern int link(string oldpath, string newpath);
|
|
|
|
[DllImport("libc", SetLastError = true)]
|
|
private static extern int stat(string pathname, out Stat statbuf);
|
|
|
|
// ReSharper disable once InconsistentNaming
|
|
// Struct layout is likely non-portable across unices. Tread with caution.
|
|
[StructLayout(LayoutKind.Sequential)]
|
|
private struct Stat
|
|
{
|
|
public readonly long st_dev;
|
|
public readonly long st_ino;
|
|
public readonly long st_nlink;
|
|
public readonly int st_mode;
|
|
public readonly int st_uid;
|
|
public readonly int st_gid;
|
|
public readonly long st_rdev;
|
|
public readonly long st_size;
|
|
public readonly long st_blksize;
|
|
public readonly long st_blocks;
|
|
public readonly Timespec st_atim;
|
|
public readonly Timespec st_mtim;
|
|
public readonly Timespec st_ctim;
|
|
}
|
|
|
|
// ReSharper disable once InconsistentNaming
|
|
[StructLayout(LayoutKind.Sequential)]
|
|
private struct Timespec
|
|
{
|
|
public readonly long tv_sec;
|
|
public readonly long tv_nsec;
|
|
}
|
|
|
|
#pragma warning restore IDE1006
|
|
|
|
#endregion
|
|
}
|
|
}
|