2021-07-06 11:16:17 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Max Wipfli <max.wipfli@serenityos.org>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
2021-11-10 11:05:21 +01:00
|
|
|
#include <AK/Vector.h>
|
2021-07-06 11:16:17 +02:00
|
|
|
#include <Kernel/KLexicalPath.h>
|
|
|
|
|
|
|
|
namespace Kernel::KLexicalPath {
|
|
|
|
|
2021-07-11 13:00:51 +02:00
|
|
|
static StringView const s_single_dot = "."sv;
|
|
|
|
|
2021-11-11 00:55:02 +01:00
|
|
|
bool is_absolute(StringView path)
|
2021-07-06 11:16:17 +02:00
|
|
|
{
|
|
|
|
return !path.is_empty() && path[0] == '/';
|
|
|
|
}
|
|
|
|
|
2021-11-11 00:55:02 +01:00
|
|
|
bool is_canonical(StringView path)
|
2021-07-06 11:16:17 +02:00
|
|
|
{
|
|
|
|
// FIXME: This can probably be done more efficiently.
|
|
|
|
if (path.is_empty())
|
|
|
|
return false;
|
|
|
|
if (path.ends_with('/') && path.length() != 1)
|
|
|
|
return false;
|
|
|
|
if (path.starts_with("./"sv) || path.contains("/./"sv) || path.ends_with("/."sv))
|
|
|
|
return false;
|
|
|
|
if (path.starts_with("../"sv) || path.contains("/../"sv) || path.ends_with("/.."))
|
|
|
|
return false;
|
|
|
|
if (path.contains("//"sv))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-11-11 00:55:02 +01:00
|
|
|
StringView basename(StringView a_path)
|
2021-07-06 11:16:17 +02:00
|
|
|
{
|
2021-07-11 13:00:51 +02:00
|
|
|
if (a_path == "/"sv)
|
|
|
|
return a_path;
|
|
|
|
if (a_path.is_empty())
|
|
|
|
return s_single_dot;
|
|
|
|
auto path = a_path.trim("/"sv, TrimMode::Right);
|
|
|
|
// NOTE: If it's empty now, it means the path was just a series of slashes.
|
|
|
|
if (path.is_empty())
|
|
|
|
return a_path.substring_view(0, 1);
|
2021-07-06 11:16:17 +02:00
|
|
|
auto slash_index = path.find_last('/');
|
2021-07-11 13:00:51 +02:00
|
|
|
if (!slash_index.has_value())
|
2021-07-06 11:16:17 +02:00
|
|
|
return path;
|
|
|
|
auto basename = path.substring_view(*slash_index + 1);
|
|
|
|
return basename;
|
|
|
|
}
|
|
|
|
|
2021-11-11 00:55:02 +01:00
|
|
|
StringView dirname(StringView path)
|
2021-07-06 11:16:17 +02:00
|
|
|
{
|
|
|
|
VERIFY(is_canonical(path));
|
|
|
|
auto slash_index = path.find_last('/');
|
|
|
|
VERIFY(slash_index.has_value());
|
|
|
|
return path.substring_view(0, *slash_index);
|
|
|
|
}
|
|
|
|
|
2021-11-11 00:55:02 +01:00
|
|
|
Vector<StringView> parts(StringView path)
|
2021-07-06 11:16:17 +02:00
|
|
|
{
|
|
|
|
VERIFY(is_canonical(path));
|
|
|
|
return path.split_view('/');
|
|
|
|
}
|
|
|
|
|
2021-11-11 00:55:02 +01:00
|
|
|
ErrorOr<NonnullOwnPtr<KString>> try_join(StringView first, StringView second)
|
2021-07-06 12:05:50 +02:00
|
|
|
{
|
|
|
|
VERIFY(is_canonical(first));
|
|
|
|
VERIFY(is_canonical(second));
|
|
|
|
VERIFY(!is_absolute(second));
|
|
|
|
|
|
|
|
if (first == "/"sv) {
|
|
|
|
char* buffer;
|
2021-09-06 19:24:54 +02:00
|
|
|
auto string = TRY(KString::try_create_uninitialized(1 + second.length(), buffer));
|
2021-07-06 12:05:50 +02:00
|
|
|
buffer[0] = '/';
|
|
|
|
__builtin_memcpy(buffer + 1, second.characters_without_null_termination(), second.length());
|
|
|
|
buffer[string->length()] = 0;
|
|
|
|
return string;
|
|
|
|
}
|
2021-09-06 19:24:54 +02:00
|
|
|
char* buffer;
|
|
|
|
auto string = TRY(KString::try_create_uninitialized(first.length() + 1 + second.length(), buffer));
|
|
|
|
__builtin_memcpy(buffer, first.characters_without_null_termination(), first.length());
|
|
|
|
buffer[first.length()] = '/';
|
|
|
|
__builtin_memcpy(buffer + first.length() + 1, second.characters_without_null_termination(), second.length());
|
|
|
|
buffer[string->length()] = 0;
|
|
|
|
return string;
|
2021-07-06 12:05:50 +02:00
|
|
|
}
|
|
|
|
|
2021-07-06 11:16:17 +02:00
|
|
|
}
|