Traits.cxx 4.69 KB
Newer Older
Max Kellermann's avatar
Max Kellermann committed
1
/*
2
 * Copyright 2003-2018 The Music Player Daemon Project
Max Kellermann's avatar
Max Kellermann committed
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
 * http://www.musicpd.org
 *
 * 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 2 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, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

#include "Traits.hxx"
21
#include "util/StringCompare.hxx"
Max Kellermann's avatar
Max Kellermann committed
22 23 24

#include <string.h>

25 26
template<typename Traits>
typename Traits::string
27 28
BuildPathImpl(typename Traits::const_pointer_type a, size_t a_size,
	      typename Traits::const_pointer_type b, size_t b_size)
29 30 31 32 33
{
	assert(a != nullptr);
	assert(b != nullptr);

	if (a_size == 0)
34
		return typename Traits::string(b, b_size);
35
	if (b_size == 0)
36
		return typename Traits::string(a, a_size);
37

38
	typename Traits::string result(a, a_size);
39

40 41
	if (!Traits::IsSeparator(a[a_size - 1]))
		result.push_back(Traits::SEPARATOR);
42

43
	if (Traits::IsSeparator(b[0]))
44 45 46 47 48 49 50
		result.append(b + 1, b_size - 1);
	else
		result.append(b, b_size);

	return result;
}

51
template<typename Traits>
52 53
typename Traits::const_pointer_type
GetBasePathImpl(typename Traits::const_pointer_type p)
Max Kellermann's avatar
Max Kellermann committed
54
{
55 56
#if !CLANG_CHECK_VERSION(3,6)
	/* disabled on clang due to -Wtautological-pointer-compare */
Max Kellermann's avatar
Max Kellermann committed
57
	assert(p != nullptr);
58
#endif
Max Kellermann's avatar
Max Kellermann committed
59

60
	typename Traits::const_pointer_type sep = Traits::FindLastSeparator(p);
61 62
	return sep != nullptr
		? sep + 1
Max Kellermann's avatar
Max Kellermann committed
63 64 65
		: p;
}

66 67
template<typename Traits>
typename Traits::string
68
GetParentPathImpl(typename Traits::const_pointer_type p)
Max Kellermann's avatar
Max Kellermann committed
69
{
70 71
#if !CLANG_CHECK_VERSION(3,6)
	/* disabled on clang due to -Wtautological-pointer-compare */
Max Kellermann's avatar
Max Kellermann committed
72
	assert(p != nullptr);
73
#endif
Max Kellermann's avatar
Max Kellermann committed
74

75
	typename Traits::const_pointer_type sep = Traits::FindLastSeparator(p);
76
	if (sep == nullptr)
77
		return typename Traits::string(Traits::CURRENT_DIRECTORY);
78 79
	if (sep == p)
		return typename Traits::string(p, p + 1);
80
#ifdef _WIN32
81 82 83
	if (Traits::IsDrive(p) && sep == p + 2)
		return typename Traits::string(p, p + 3);
#endif
84 85 86
	return typename Traits::string(p, sep);
}

87
template<typename Traits>
88 89 90
typename Traits::const_pointer_type
RelativePathImpl(typename Traits::const_pointer_type base,
		 typename Traits::const_pointer_type other)
91 92 93 94
{
	assert(base != nullptr);
	assert(other != nullptr);

95 96
	other = StringAfterPrefix(other, base);
	if (other == nullptr)
97 98 99
		/* mismatch */
		return nullptr;

100
	if (*other != 0) {
101 102 103 104 105 106 107
		if (!Traits::IsSeparator(*other)) {
			if (*base != 0 && Traits::IsSeparator(other[-1]))
				/* "other" has no more slash, but the
				   matching base ended with a slash:
				   enough to detect a match */
				return other;

108 109
			/* mismatch */
			return nullptr;
110
		}
111 112 113 114 115 116 117 118 119 120

		/* skip remaining path separators */
		do {
			++other;
		} while (Traits::IsSeparator(*other));
	}

	return other;
}

121
PathTraitsFS::string
122
PathTraitsFS::Build(const_pointer_type a, size_t a_size,
123
		    const_pointer_type b, size_t b_size) noexcept
124 125 126 127
{
	return BuildPathImpl<PathTraitsFS>(a, a_size, b, b_size);
}

128
PathTraitsFS::const_pointer_type
129
PathTraitsFS::GetBase(PathTraitsFS::const_pointer_type p) noexcept
130 131 132 133 134
{
	return GetBasePathImpl<PathTraitsFS>(p);
}

PathTraitsFS::string
135
PathTraitsFS::GetParent(PathTraitsFS::const_pointer_type p) noexcept
136 137 138 139
{
	return GetParentPathImpl<PathTraitsFS>(p);
}

140
PathTraitsFS::const_pointer_type
141
PathTraitsFS::Relative(const_pointer_type base, const_pointer_type other) noexcept
142 143 144 145
{
	return RelativePathImpl<PathTraitsFS>(base, other);
}

146 147 148 149 150 151 152 153 154 155 156 157 158 159
PathTraitsFS::string
PathTraitsFS::Apply(const_pointer_type base, const_pointer_type path) noexcept
{
	// TODO: support the Windows syntax (absolute path with or without drive, drive with relative path)

	if (base == nullptr)
		return path;

	if (IsAbsolute(path))
		return path;

	return Build(base, path);
}

160
PathTraitsUTF8::string
161
PathTraitsUTF8::Build(const_pointer_type a, size_t a_size,
162
		      const_pointer_type b, size_t b_size) noexcept
163 164 165 166
{
	return BuildPathImpl<PathTraitsUTF8>(a, a_size, b, b_size);
}

167
PathTraitsUTF8::const_pointer_type
168
PathTraitsUTF8::GetBase(const_pointer_type p) noexcept
169 170 171 172 173
{
	return GetBasePathImpl<PathTraitsUTF8>(p);
}

PathTraitsUTF8::string
174
PathTraitsUTF8::GetParent(const_pointer_type p) noexcept
175 176
{
	return GetParentPathImpl<PathTraitsUTF8>(p);
Max Kellermann's avatar
Max Kellermann committed
177
}
178

179
PathTraitsUTF8::const_pointer_type
180 181
PathTraitsUTF8::Relative(const_pointer_type base,
			 const_pointer_type other) noexcept
182 183 184
{
	return RelativePathImpl<PathTraitsUTF8>(base, other);
}