summaryrefslogtreecommitdiff
path: root/shared/longPathPrefix.cpp
blob: 9ce74c8b9f2130ee071271a547ed7430b6850806 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
#include "longPathPrefix.h"
#include <boost/scoped_array.hpp>
#include <wx/msw/wrapwin.h> //includes "windows.h"
#include "fileError.h"
#include "systemFunctions.h"
#include "stringConv.h"
#include <wx/intl.h>

namespace
{
Zstring getFullPathName(const Zstring& relativeName, size_t proposedBufferSize = 1000)
{
    using namespace FreeFileSync;

    boost::scoped_array<DefaultChar> fullPath(new DefaultChar[proposedBufferSize]);
    const DWORD rv = ::GetFullPathName(
                         relativeName.c_str(), //__in   LPCTSTR lpFileName,
                         proposedBufferSize,   //__in   DWORD nBufferLength,
                         fullPath.get(),       //__out  LPTSTR lpBuffer,
                         NULL);                //__out  LPTSTR *lpFilePart
    if (rv == 0 || rv == proposedBufferSize)
        throw FileError(wxString(_("Error resolving full path name:")) + wxT("\n\"") + zToWx(relativeName) + wxT("\"") +
                        wxT("\n\n") + FreeFileSync::getLastErrorFormatted());
    if (rv > proposedBufferSize)
        return getFullPathName(relativeName, rv);

    return fullPath.get();
}
}


Zstring FreeFileSync::resolveRelativePath(const Zstring& path) //throw()
{
    try
    {
        return getFullPathName(path);
    }
    catch (...)
    {
        return path;
    }
}


//there are two flavors of long path prefix: one for UNC paths, one for regular paths
const Zstring LONG_PATH_PREFIX = DefaultStr("\\\\?\\");
const Zstring LONG_PATH_PREFIX_UNC = DefaultStr("\\\\?\\UNC");

template <size_t max_path>
inline
Zstring applyLongPathPrefixImpl(const Zstring& path)
{
    if (    path.length() >= max_path &&    //maximum allowed path length without prefix is (MAX_PATH - 1)
            !path.StartsWith(LONG_PATH_PREFIX))
    {
        if (path.StartsWith(DefaultStr("\\\\"))) //UNC-name, e.g. \\zenju-pc\Users
            return LONG_PATH_PREFIX_UNC + path.AfterFirst(DefaultChar('\\')); //convert to \\?\UNC\zenju-pc\Users
        else
            return LONG_PATH_PREFIX + path; //prepend \\?\ prefix
    }

    //fallback
    return path;
}


Zstring FreeFileSync::applyLongPathPrefix(const Zstring& path)
{
    return applyLongPathPrefixImpl<MAX_PATH>(path);
}


Zstring FreeFileSync::applyLongPathPrefixCreateDir(const Zstring& path) //throw()
{
    //special rule for ::CreateDirectoryEx(): MAX_PATH - 12(=^ 8.3 filename) is threshold
    return applyLongPathPrefixImpl<MAX_PATH - 12>(path);
}


Zstring FreeFileSync::removeLongPathPrefix(const Zstring& path) //throw()
{
    if (path.StartsWith(LONG_PATH_PREFIX))
    {
        Zstring finalPath = path;
        if (path.StartsWith(LONG_PATH_PREFIX_UNC)) //UNC-name
            finalPath.Replace(LONG_PATH_PREFIX_UNC, DefaultStr("\\"), false);
        else
            finalPath.Replace(LONG_PATH_PREFIX, DefaultStr(""), false);
        return finalPath;
    }

    //fallback
    return path;
}

bgstack15