Traits.cxx 4.71 KB
Newer Older
Max Kellermann's avatar
Max Kellermann committed
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2017 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 21
 * 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 "config.h"
#include "Traits.hxx"
22
#include "util/StringCompare.hxx"
Max Kellermann's avatar
Max Kellermann committed
23 24 25

#include <string.h>

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

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

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

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

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

	return result;
}

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

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

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

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

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

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

101
	if (*other != 0) {
102 103 104 105 106 107 108
		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;

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

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

	return other;
}

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

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

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

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

147 148 149 150 151 152 153 154 155 156 157 158 159 160
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);
}

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

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

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

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