c# - как извлечь образ диска FAT?

Я на самом деле пытаюсь извлечь образ диска Fat с помощью DiskUtils, но я не получаю правильные имена файлов...

Я получаю " TURNER~3TOPPER~1.P~1 "вместо" TURNEROVERTOPPERSHEATH.PPTX-файл"

FatFileSystem FatImg = new FatFileSystem(MS); //MS = Fat Image MemoryStream
foreach(DiscDirectoryInfo Di in FatImg.Root.GetDirectories())
{
    foreach(DiscFileInfo Fi in Di.GetFiles())
    {
        Stream St = Fi.OpenRead(); // Correct Stream
        string FName = Fi.Name; //Wrong Name
    }
}

Это потому, что DiscUtils не поддерживает LFN [длинные имена файлов]...

поэтому я ищу идеальную библиотеку для извлечения этих файлов, прежде чем я попытаюсь сделать это сам...

есть ли способ извлечь его [возможно, с помощью DiscUtils] без ошибок имени файла...

5 ответов


вот некоторые изменения, которые вы можете добавить к DiscUtils в поддержку FAT LFNs:

во-первых, внесите эти изменения в Fat\Directory.cs файл, как это (вам нужно добавить _lfns переменной GetLfnChunk функция и изменить существующий LoadEntries функция для добавления строк, отмеченных //+++ ниже):

internal Dictionary<string, string> _lfns = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);

private static string GetLfnChunk(byte[] buffer)
{
    // see http://home.teleport.com/~brainy/lfn.htm
    // NOTE: we assume ordinals are ok here.
    char[] chars = new char[13];
    chars[0] = (char)(256 * buffer[2] + buffer[1]);
    chars[1] = (char)(256 * buffer[4] + buffer[3]);
    chars[2] = (char)(256 * buffer[6] + buffer[5]);
    chars[3] = (char)(256 * buffer[8] + buffer[7]);
    chars[4] = (char)(256 * buffer[10] + buffer[9]);

    chars[5] = (char)(256 * buffer[15] + buffer[14]);
    chars[6] = (char)(256 * buffer[17] + buffer[16]);
    chars[7] = (char)(256 * buffer[19] + buffer[18]);
    chars[8] = (char)(256 * buffer[21] + buffer[20]);
    chars[9] = (char)(256 * buffer[23] + buffer[22]);
    chars[10] = (char)(256 * buffer[25] + buffer[24]);

    chars[11] = (char)(256 * buffer[29] + buffer[28]);
    chars[12] = (char)(256 * buffer[31] + buffer[30]);
    string chunk = new string(chars);
    int zero = chunk.IndexOf('');
    return zero >= 0 ? chunk.Substring(0, zero) : chunk;
}

private void LoadEntries()
{
    _entries = new Dictionary<long, DirectoryEntry>();
    _freeEntries = new List<long>();

    _selfEntryLocation = -1;
    _parentEntryLocation = -1;

    string lfn = null;  //+++
    while (_dirStream.Position < _dirStream.Length)
    {
        long streamPos = _dirStream.Position;
        DirectoryEntry entry = new DirectoryEntry(_fileSystem.FatOptions, _dirStream);

        if (entry.Attributes == (FatAttributes.ReadOnly | FatAttributes.Hidden | FatAttributes.System | FatAttributes.VolumeId))
        {
            // Long File Name entry
            _dirStream.Position = streamPos;  //+++
            lfn = GetLfnChunk(Utilities.ReadFully(_dirStream, 32)) + lfn;  //+++
        }
        else if (entry.Name.IsDeleted())
        {
            // E5 = Free Entry
            _freeEntries.Add(streamPos);
            lfn = null; //+++
        }
        else if (entry.Name == FileName.SelfEntryName)
        {
            _selfEntry = entry;
            _selfEntryLocation = streamPos;
            lfn = null; //+++
        }
        else if (entry.Name == FileName.ParentEntryName)
        {
            _parentEntry = entry;
            _parentEntryLocation = streamPos;
            lfn = null; //+++
        }
        else if (entry.Name == FileName.Null)
        {
            // Free Entry, no more entries available
            _endOfEntries = streamPos;
            lfn = null; //+++
            break;
        }
        else
        {
            if (lfn != null) //+++
            { //+++
                _lfns.Add(entry.Name.GetDisplayName(_fileSystem.FatOptions.FileNameEncoding), lfn); //+++
            } //+++
            _entries.Add(streamPos, entry);
            lfn = null; //+++
        }
    }
}

во-вторых, добавьте эти две публичные функции в . Они будут новыми API для запроса на LFNs:

/// <summary>
/// Gets the long name of a given file.
/// </summary>
/// <param name="shortFullPath">The short full path to the file. Input path segments must be short names.</param>
/// <returns>The corresponding long file name.</returns>
public string GetLongFileName(string shortFullPath)
{
    if (shortFullPath == null)
        throw new ArgumentNullException("shortFullPath");

    string dirPath = Path.GetDirectoryName(shortFullPath);
    string fileName = Path.GetFileName(shortFullPath);
    Directory dir = GetDirectory(dirPath);
    if (dir == null)
        return fileName;

    string lfn;
    if (dir._lfns.TryGetValue(Path.GetFileName(shortFullPath), out lfn))
        return lfn;

    return fileName;
}

/// <summary>
/// Gets the long path to a given file.
/// </summary>
/// <param name="shortFullPath">The short full path to the file. Input path segments must be short names.</param>
/// <returns>The corresponding long file path to the file or null if not found.</returns>
public string GetLongFilePath(string shortFullPath)
{
    if (shortFullPath == null)
        throw new ArgumentNullException("shortFullPath");

    string path = null;
    string current = null;
    foreach (string segment in shortFullPath.Split(Path.DirectorySeparatorChar))
    {
        if (current == null)
        {
            current = segment;
            path = GetLongFileName(current);
        }
        else
        {
            current = Path.Combine(current, segment);
            path = Path.Combine(path, GetLongFileName(current));
        }
    }
    return path;
}

и это все. Теперь вы сможете сбросить весь жирный диск рекурсивно, например:

static void Main(string[] args)
{
    using (FileStream fs = File.Open("fat.ima", FileMode.Open))
    {
        using (FatFileSystem floppy = new FatFileSystem(fs))
        {
            Dump(floppy.Root);
        }
    }
}

static void Dump(DiscDirectoryInfo di)
{
    foreach (DiscDirectoryInfo subdi in di.GetDirectories())
    {
        Dump(subdi);
    }
    foreach (DiscFileInfo fi in di.GetFiles())
    {
        Console.WriteLine(fi.FullName);
        // get LFN name
        Console.WriteLine(" " + ((FatFileSystem)di.FileSystem).GetLongFileName(fi.FullName));


        // get LFN-ed full path
        Console.WriteLine(" " + ((FatFileSystem)di.FileSystem).GetLongFilePath(fi.FullName));
    }
}

используйте на свой страх и риск! :)


7-Zip смогите извлечь тучные изображения. Существует библиотека оболочки C# в http://sevenzipsharp.codeplex.com который может читать имена файлов и извлекать в потоки.


казалось бы, что в DiscUtils нет поддержки файлов с длинными именами для FAT. проверить этот пост. который, я уверен, вы знаете, так как похоже, что вы задали вопрос.


лучший метод с командной строкой-использовать это:

7z " e "+ источник + "- o "+ путь + "* - r"

  • " e " = извлечь
  • source = путь к образу диска
  • "- o " = вывод / извлечение в
  • path = пункт назначения для извлечения в
  • " *" = все файлы
  • " - r " = рекурсивный

В C# вы можете использовать этот метод, который я сделал (я скопировал 7z.exe в папку отладки my приложение)

public void ExtractDiskImage(string pathToDiskImage, string extractPath, bool WaitForFinish)
{
    ProcessStartInfo UnzipDiskImage = new ProcessStartInfo("7z.exe");
    StringBuilder str = new StringBuilder();
    str.Append("e ");
    str.Append(pathToDiskImage);
    str.Append(" -o");
    str.Append(extractPath);
    str.Append(" * -r");
    UnzipDiskImage.Arguments = str.ToString();
    UnzipDiskImage.WindowStyle = ProcessWindowStyle.Hidden;
    Process process = Process.Start(UnzipDiskImage);
    if(WaitForFinish == true)
    {
        process.WaitForExit(); //My app had to wait for the extract to finish
    }
}

команда DOS DIR с переключателем /X отображает как длинные, так и короткие имена.

в противном случае есть утилита для сохранения LFNs:DOSLFNBk.exe это поможет в создании таблицы сопоставления.

Я не знаю ответов, которые вы искали, но не хватает ручного создания таблицы сначала, чтобы вы могли сделать отображение без DiskUtil, я не могу придумать утилиту или метод для достижения вашей цели-но, как вы упоминаете, здесь так кто-то может знать альтернативу.

Кевин из DiskUtils упомянул, что он будет нарушать патент Mirosoft, если он включил поддержку LFN, я не предлагаю Вам нарушать либо (определенно не его коммерческий проект), но если это просто личное или если вы можете найти библиотеку, такую как 7-Zip, имеет лицензию...

ваш скриншот показывает VNext.. та же ошибка с версиями RTM?