2015-05-20 20:37:07 -07:00
|
|
|
// Copyright 2015 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-08-02 09:45:56 -07:00
|
|
|
#include <algorithm>
|
2015-06-21 07:11:32 -07:00
|
|
|
#include <iterator>
|
2015-05-20 20:37:07 -07:00
|
|
|
#include "common/assert.h"
|
2017-05-21 00:11:36 -07:00
|
|
|
#include "core/hle/kernel/errors.h"
|
2016-09-20 23:52:38 -07:00
|
|
|
#include "core/hle/kernel/vm_manager.h"
|
2016-05-26 10:53:30 -07:00
|
|
|
#include "core/memory.h"
|
2016-01-30 10:41:04 -08:00
|
|
|
#include "core/mmio.h"
|
2015-05-20 20:37:07 -07:00
|
|
|
|
|
|
|
namespace Kernel {
|
|
|
|
|
2015-07-17 17:55:48 -07:00
|
|
|
static const char* GetMemoryStateName(MemoryState state) {
|
|
|
|
static const char* names[] = {
|
2016-09-17 17:38:01 -07:00
|
|
|
"Free", "Reserved", "IO", "Static", "Code", "Private",
|
|
|
|
"Shared", "Continuous", "Aliased", "Alias", "AliasCode", "Locked",
|
2015-07-17 17:55:48 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
return names[(int)state];
|
|
|
|
}
|
|
|
|
|
2015-05-20 20:37:07 -07:00
|
|
|
bool VirtualMemoryArea::CanBeMergedWith(const VirtualMemoryArea& next) const {
|
|
|
|
ASSERT(base + size == next.base);
|
2016-09-17 17:38:01 -07:00
|
|
|
if (permissions != next.permissions || meminfo_state != next.meminfo_state ||
|
|
|
|
type != next.type) {
|
2015-05-20 20:37:07 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (type == VMAType::BackingMemory && backing_memory + size != next.backing_memory) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (type == VMAType::MMIO && paddr + size != next.paddr) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-12-10 19:01:09 -08:00
|
|
|
VMManager::VMManager(Memory::MemorySystem& memory) : memory(memory) {
|
2015-05-20 20:37:07 -07:00
|
|
|
Reset();
|
|
|
|
}
|
|
|
|
|
2015-07-09 18:52:15 -07:00
|
|
|
VMManager::~VMManager() {
|
|
|
|
Reset();
|
|
|
|
}
|
|
|
|
|
2015-05-20 20:37:07 -07:00
|
|
|
void VMManager::Reset() {
|
|
|
|
vma_map.clear();
|
|
|
|
|
|
|
|
// Initialize the map with a single free region covering the entire managed space.
|
|
|
|
VirtualMemoryArea initial_vma;
|
|
|
|
initial_vma.size = MAX_ADDRESS;
|
|
|
|
vma_map.emplace(initial_vma.base, initial_vma);
|
|
|
|
|
2017-07-21 19:17:57 -07:00
|
|
|
page_table.pointers.fill(nullptr);
|
|
|
|
page_table.attributes.fill(Memory::PageType::Unmapped);
|
|
|
|
|
2015-05-20 20:37:07 -07:00
|
|
|
UpdatePageTableForVMA(initial_vma);
|
|
|
|
}
|
|
|
|
|
|
|
|
VMManager::VMAHandle VMManager::FindVMA(VAddr target) const {
|
2015-07-17 19:19:16 -07:00
|
|
|
if (target >= MAX_ADDRESS) {
|
|
|
|
return vma_map.end();
|
|
|
|
} else {
|
|
|
|
return std::prev(vma_map.upper_bound(target));
|
|
|
|
}
|
2015-05-20 20:37:07 -07:00
|
|
|
}
|
|
|
|
|
2018-11-08 11:05:54 -08:00
|
|
|
ResultVal<VAddr> VMManager::MapBackingMemoryToBase(VAddr base, u32 region_size, u8* memory,
|
|
|
|
u32 size, MemoryState state) {
|
2017-11-07 10:38:33 -08:00
|
|
|
|
|
|
|
// Find the first Free VMA.
|
|
|
|
VMAHandle vma_handle = std::find_if(vma_map.begin(), vma_map.end(), [&](const auto& vma) {
|
|
|
|
if (vma.second.type != VMAType::Free)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
VAddr vma_end = vma.second.base + vma.second.size;
|
|
|
|
return vma_end > base && vma_end >= base + size;
|
|
|
|
});
|
|
|
|
|
2017-11-07 11:35:17 -08:00
|
|
|
VAddr target = std::max(base, vma_handle->second.base);
|
|
|
|
|
|
|
|
// Do not try to allocate the block if there are no available addresses within the desired
|
|
|
|
// region.
|
|
|
|
if (vma_handle == vma_map.end() || target + size > base + region_size) {
|
2017-11-07 10:38:33 -08:00
|
|
|
return ResultCode(ErrorDescription::OutOfMemory, ErrorModule::Kernel,
|
|
|
|
ErrorSummary::OutOfResource, ErrorLevel::Permanent);
|
|
|
|
}
|
|
|
|
|
2018-11-08 11:05:54 -08:00
|
|
|
auto result = MapBackingMemory(target, memory, size, state);
|
2017-11-07 10:38:33 -08:00
|
|
|
|
|
|
|
if (result.Failed())
|
|
|
|
return result.Code();
|
|
|
|
|
|
|
|
return MakeResult<VAddr>(target);
|
|
|
|
}
|
|
|
|
|
2016-09-17 17:38:01 -07:00
|
|
|
ResultVal<VMManager::VMAHandle> VMManager::MapBackingMemory(VAddr target, u8* memory, u32 size,
|
|
|
|
MemoryState state) {
|
2015-05-20 20:37:07 -07:00
|
|
|
ASSERT(memory != nullptr);
|
|
|
|
|
|
|
|
// This is the appropriately sized VMA that will turn into our allocation.
|
|
|
|
CASCADE_RESULT(VMAIter vma_handle, CarveVMA(target, size));
|
|
|
|
VirtualMemoryArea& final_vma = vma_handle->second;
|
|
|
|
ASSERT(final_vma.size == size);
|
|
|
|
|
|
|
|
final_vma.type = VMAType::BackingMemory;
|
|
|
|
final_vma.permissions = VMAPermission::ReadWrite;
|
|
|
|
final_vma.meminfo_state = state;
|
|
|
|
final_vma.backing_memory = memory;
|
|
|
|
UpdatePageTableForVMA(final_vma);
|
|
|
|
|
|
|
|
return MakeResult<VMAHandle>(MergeAdjacent(vma_handle));
|
|
|
|
}
|
|
|
|
|
2016-09-17 17:38:01 -07:00
|
|
|
ResultVal<VMManager::VMAHandle> VMManager::MapMMIO(VAddr target, PAddr paddr, u32 size,
|
|
|
|
MemoryState state,
|
|
|
|
Memory::MMIORegionPointer mmio_handler) {
|
2015-05-20 20:37:07 -07:00
|
|
|
// This is the appropriately sized VMA that will turn into our allocation.
|
|
|
|
CASCADE_RESULT(VMAIter vma_handle, CarveVMA(target, size));
|
|
|
|
VirtualMemoryArea& final_vma = vma_handle->second;
|
|
|
|
ASSERT(final_vma.size == size);
|
|
|
|
|
|
|
|
final_vma.type = VMAType::MMIO;
|
|
|
|
final_vma.permissions = VMAPermission::ReadWrite;
|
|
|
|
final_vma.meminfo_state = state;
|
|
|
|
final_vma.paddr = paddr;
|
2016-01-30 10:41:04 -08:00
|
|
|
final_vma.mmio_handler = mmio_handler;
|
2015-05-20 20:37:07 -07:00
|
|
|
UpdatePageTableForVMA(final_vma);
|
|
|
|
|
|
|
|
return MakeResult<VMAHandle>(MergeAdjacent(vma_handle));
|
|
|
|
}
|
|
|
|
|
2017-11-19 08:24:23 -08:00
|
|
|
ResultCode VMManager::ChangeMemoryState(VAddr target, u32 size, MemoryState expected_state,
|
|
|
|
VMAPermission expected_perms, MemoryState new_state,
|
|
|
|
VMAPermission new_perms) {
|
|
|
|
VAddr target_end = target + size;
|
|
|
|
VMAIter begin_vma = StripIterConstness(FindVMA(target));
|
|
|
|
VMAIter i_end = vma_map.lower_bound(target_end);
|
|
|
|
|
|
|
|
if (begin_vma == vma_map.end())
|
|
|
|
return ERR_INVALID_ADDRESS;
|
|
|
|
|
|
|
|
for (auto i = begin_vma; i != i_end; ++i) {
|
|
|
|
auto& vma = i->second;
|
|
|
|
if (vma.meminfo_state != expected_state) {
|
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
u32 perms = static_cast<u32>(expected_perms);
|
|
|
|
if ((static_cast<u32>(vma.permissions) & perms) != perms) {
|
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CASCADE_RESULT(auto vma, CarveVMARange(target, size));
|
|
|
|
ASSERT(vma->second.size == size);
|
|
|
|
|
|
|
|
vma->second.permissions = new_perms;
|
|
|
|
vma->second.meminfo_state = new_state;
|
|
|
|
UpdatePageTableForVMA(vma->second);
|
|
|
|
|
|
|
|
MergeAdjacent(vma);
|
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-07-17 19:19:16 -07:00
|
|
|
VMManager::VMAIter VMManager::Unmap(VMAIter vma_handle) {
|
|
|
|
VirtualMemoryArea& vma = vma_handle->second;
|
2015-05-20 20:37:07 -07:00
|
|
|
vma.type = VMAType::Free;
|
|
|
|
vma.permissions = VMAPermission::None;
|
|
|
|
vma.meminfo_state = MemoryState::Free;
|
|
|
|
|
|
|
|
vma.backing_memory = nullptr;
|
|
|
|
vma.paddr = 0;
|
|
|
|
|
|
|
|
UpdatePageTableForVMA(vma);
|
|
|
|
|
2015-07-17 19:19:16 -07:00
|
|
|
return MergeAdjacent(vma_handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
ResultCode VMManager::UnmapRange(VAddr target, u32 size) {
|
|
|
|
CASCADE_RESULT(VMAIter vma, CarveVMARange(target, size));
|
2018-08-02 09:19:05 -07:00
|
|
|
const VAddr target_end = target + size;
|
2015-07-17 19:19:16 -07:00
|
|
|
|
2018-08-02 09:19:05 -07:00
|
|
|
const VMAIter end = vma_map.end();
|
2015-07-17 19:19:16 -07:00
|
|
|
// The comparison against the end of the range must be done using addresses since VMAs can be
|
|
|
|
// merged during this process, causing invalidation of the iterators.
|
|
|
|
while (vma != end && vma->second.base < target_end) {
|
|
|
|
vma = std::next(Unmap(vma));
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(FindVMA(target)->second.size >= size);
|
|
|
|
return RESULT_SUCCESS;
|
2015-05-20 20:37:07 -07:00
|
|
|
}
|
|
|
|
|
2015-07-17 19:19:16 -07:00
|
|
|
VMManager::VMAHandle VMManager::Reprotect(VMAHandle vma_handle, VMAPermission new_perms) {
|
2015-05-20 20:37:07 -07:00
|
|
|
VMAIter iter = StripIterConstness(vma_handle);
|
|
|
|
|
|
|
|
VirtualMemoryArea& vma = iter->second;
|
|
|
|
vma.permissions = new_perms;
|
|
|
|
UpdatePageTableForVMA(vma);
|
|
|
|
|
2015-07-17 19:19:16 -07:00
|
|
|
return MergeAdjacent(iter);
|
|
|
|
}
|
|
|
|
|
|
|
|
ResultCode VMManager::ReprotectRange(VAddr target, u32 size, VMAPermission new_perms) {
|
|
|
|
CASCADE_RESULT(VMAIter vma, CarveVMARange(target, size));
|
2018-08-02 09:19:05 -07:00
|
|
|
const VAddr target_end = target + size;
|
2015-07-17 19:19:16 -07:00
|
|
|
|
2018-08-02 09:19:05 -07:00
|
|
|
const VMAIter end = vma_map.end();
|
2015-07-17 19:19:16 -07:00
|
|
|
// The comparison against the end of the range must be done using addresses since VMAs can be
|
|
|
|
// merged during this process, causing invalidation of the iterators.
|
|
|
|
while (vma != end && vma->second.base < target_end) {
|
|
|
|
vma = std::next(StripIterConstness(Reprotect(vma, new_perms)));
|
|
|
|
}
|
|
|
|
|
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-07-17 17:55:48 -07:00
|
|
|
void VMManager::LogLayout(Log::Level log_level) const {
|
2015-07-09 18:52:15 -07:00
|
|
|
for (const auto& p : vma_map) {
|
|
|
|
const VirtualMemoryArea& vma = p.second;
|
2018-06-29 04:18:07 -07:00
|
|
|
LOG_GENERIC(::Log::Class::Kernel, log_level, "{:08X} - {:08X} size: {:8X} {}{}{} {}",
|
2018-06-29 06:56:12 -07:00
|
|
|
vma.base, vma.base + vma.size, vma.size,
|
|
|
|
(u8)vma.permissions & (u8)VMAPermission::Read ? 'R' : '-',
|
|
|
|
(u8)vma.permissions & (u8)VMAPermission::Write ? 'W' : '-',
|
|
|
|
(u8)vma.permissions & (u8)VMAPermission::Execute ? 'X' : '-',
|
|
|
|
GetMemoryStateName(vma.meminfo_state));
|
2015-07-09 18:52:15 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-17 17:38:01 -07:00
|
|
|
VMManager::VMAIter VMManager::StripIterConstness(const VMAHandle& iter) {
|
2015-05-20 20:37:07 -07:00
|
|
|
// This uses a neat C++ trick to convert a const_iterator to a regular iterator, given
|
|
|
|
// non-const access to its container.
|
|
|
|
return vma_map.erase(iter, iter); // Erases an empty range of elements
|
|
|
|
}
|
|
|
|
|
|
|
|
ResultVal<VMManager::VMAIter> VMManager::CarveVMA(VAddr base, u32 size) {
|
2018-03-28 07:28:55 -07:00
|
|
|
ASSERT_MSG((size & Memory::PAGE_MASK) == 0, "non-page aligned size: {:#10X}", size);
|
2018-03-27 08:28:42 -07:00
|
|
|
ASSERT_MSG((base & Memory::PAGE_MASK) == 0, "non-page aligned base: {:#010X}", base);
|
2015-05-20 20:37:07 -07:00
|
|
|
|
|
|
|
VMAIter vma_handle = StripIterConstness(FindVMA(base));
|
|
|
|
if (vma_handle == vma_map.end()) {
|
|
|
|
// Target address is outside the range managed by the kernel
|
2015-07-17 18:34:50 -07:00
|
|
|
return ERR_INVALID_ADDRESS;
|
2015-05-20 20:37:07 -07:00
|
|
|
}
|
|
|
|
|
2018-08-02 09:19:05 -07:00
|
|
|
const VirtualMemoryArea& vma = vma_handle->second;
|
2015-05-20 20:37:07 -07:00
|
|
|
if (vma.type != VMAType::Free) {
|
|
|
|
// Region is already allocated
|
2015-07-17 18:34:50 -07:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
2015-05-20 20:37:07 -07:00
|
|
|
}
|
|
|
|
|
2018-08-02 09:19:05 -07:00
|
|
|
const VAddr start_in_vma = base - vma.base;
|
|
|
|
const VAddr end_in_vma = start_in_vma + size;
|
2015-05-20 20:37:07 -07:00
|
|
|
|
|
|
|
if (end_in_vma > vma.size) {
|
|
|
|
// Requested allocation doesn't fit inside VMA
|
2015-07-17 18:34:50 -07:00
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
2015-05-20 20:37:07 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (end_in_vma != vma.size) {
|
|
|
|
// Split VMA at the end of the allocated region
|
|
|
|
SplitVMA(vma_handle, end_in_vma);
|
|
|
|
}
|
|
|
|
if (start_in_vma != 0) {
|
|
|
|
// Split VMA at the start of the allocated region
|
|
|
|
vma_handle = SplitVMA(vma_handle, start_in_vma);
|
|
|
|
}
|
|
|
|
|
|
|
|
return MakeResult<VMAIter>(vma_handle);
|
|
|
|
}
|
|
|
|
|
2015-07-17 19:19:16 -07:00
|
|
|
ResultVal<VMManager::VMAIter> VMManager::CarveVMARange(VAddr target, u32 size) {
|
2018-03-28 07:28:55 -07:00
|
|
|
ASSERT_MSG((size & Memory::PAGE_MASK) == 0, "non-page aligned size: {:#10X}", size);
|
2018-03-27 08:28:42 -07:00
|
|
|
ASSERT_MSG((target & Memory::PAGE_MASK) == 0, "non-page aligned base: {:#010X}", target);
|
2015-07-17 19:19:16 -07:00
|
|
|
|
2018-08-02 09:19:05 -07:00
|
|
|
const VAddr target_end = target + size;
|
2015-07-17 19:19:16 -07:00
|
|
|
ASSERT(target_end >= target);
|
|
|
|
ASSERT(target_end <= MAX_ADDRESS);
|
|
|
|
ASSERT(size > 0);
|
|
|
|
|
|
|
|
VMAIter begin_vma = StripIterConstness(FindVMA(target));
|
2018-08-02 09:19:05 -07:00
|
|
|
const VMAIter i_end = vma_map.lower_bound(target_end);
|
2018-08-02 09:45:56 -07:00
|
|
|
if (std::any_of(begin_vma, i_end,
|
|
|
|
[](const auto& entry) { return entry.second.type == VMAType::Free; })) {
|
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
2015-07-17 19:19:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (target != begin_vma->second.base) {
|
|
|
|
begin_vma = SplitVMA(begin_vma, target - begin_vma->second.base);
|
|
|
|
}
|
|
|
|
|
|
|
|
VMAIter end_vma = StripIterConstness(FindVMA(target_end));
|
|
|
|
if (end_vma != vma_map.end() && target_end != end_vma->second.base) {
|
|
|
|
end_vma = SplitVMA(end_vma, target_end - end_vma->second.base);
|
|
|
|
}
|
|
|
|
|
|
|
|
return MakeResult<VMAIter>(begin_vma);
|
|
|
|
}
|
|
|
|
|
2015-05-20 20:37:07 -07:00
|
|
|
VMManager::VMAIter VMManager::SplitVMA(VMAIter vma_handle, u32 offset_in_vma) {
|
|
|
|
VirtualMemoryArea& old_vma = vma_handle->second;
|
|
|
|
VirtualMemoryArea new_vma = old_vma; // Make a copy of the VMA
|
|
|
|
|
|
|
|
// For now, don't allow no-op VMA splits (trying to split at a boundary) because it's probably
|
|
|
|
// a bug. This restriction might be removed later.
|
|
|
|
ASSERT(offset_in_vma < old_vma.size);
|
|
|
|
ASSERT(offset_in_vma > 0);
|
|
|
|
|
|
|
|
old_vma.size = offset_in_vma;
|
|
|
|
new_vma.base += offset_in_vma;
|
|
|
|
new_vma.size -= offset_in_vma;
|
|
|
|
|
|
|
|
switch (new_vma.type) {
|
|
|
|
case VMAType::Free:
|
|
|
|
break;
|
|
|
|
case VMAType::BackingMemory:
|
|
|
|
new_vma.backing_memory += offset_in_vma;
|
|
|
|
break;
|
|
|
|
case VMAType::MMIO:
|
|
|
|
new_vma.paddr += offset_in_vma;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(old_vma.CanBeMergedWith(new_vma));
|
|
|
|
|
|
|
|
return vma_map.emplace_hint(std::next(vma_handle), new_vma.base, new_vma);
|
|
|
|
}
|
|
|
|
|
|
|
|
VMManager::VMAIter VMManager::MergeAdjacent(VMAIter iter) {
|
2018-08-02 09:19:05 -07:00
|
|
|
const VMAIter next_vma = std::next(iter);
|
2015-05-20 20:37:07 -07:00
|
|
|
if (next_vma != vma_map.end() && iter->second.CanBeMergedWith(next_vma->second)) {
|
|
|
|
iter->second.size += next_vma->second.size;
|
|
|
|
vma_map.erase(next_vma);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iter != vma_map.begin()) {
|
|
|
|
VMAIter prev_vma = std::prev(iter);
|
|
|
|
if (prev_vma->second.CanBeMergedWith(iter->second)) {
|
|
|
|
prev_vma->second.size += iter->second.size;
|
|
|
|
vma_map.erase(iter);
|
|
|
|
iter = prev_vma;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VMManager::UpdatePageTableForVMA(const VirtualMemoryArea& vma) {
|
|
|
|
switch (vma.type) {
|
|
|
|
case VMAType::Free:
|
2018-12-10 19:01:09 -08:00
|
|
|
memory.UnmapRegion(page_table, vma.base, vma.size);
|
2015-05-20 20:37:07 -07:00
|
|
|
break;
|
|
|
|
case VMAType::BackingMemory:
|
2018-12-10 19:01:09 -08:00
|
|
|
memory.MapMemoryRegion(page_table, vma.base, vma.size, vma.backing_memory);
|
2015-05-20 20:37:07 -07:00
|
|
|
break;
|
|
|
|
case VMAType::MMIO:
|
2018-12-10 19:01:09 -08:00
|
|
|
memory.MapIoRegion(page_table, vma.base, vma.size, vma.mmio_handler);
|
2015-05-20 20:37:07 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-11-06 12:00:47 -08:00
|
|
|
|
|
|
|
ResultVal<std::vector<std::pair<u8*, u32>>> VMManager::GetBackingBlocksForRange(VAddr address,
|
|
|
|
u32 size) {
|
|
|
|
std::vector<std::pair<u8*, u32>> backing_blocks;
|
|
|
|
VAddr interval_target = address;
|
|
|
|
while (interval_target != address + size) {
|
|
|
|
auto vma = FindVMA(interval_target);
|
|
|
|
if (vma->second.type != VMAType::BackingMemory) {
|
|
|
|
LOG_ERROR(Kernel, "Trying to use already freed memory");
|
|
|
|
return ERR_INVALID_ADDRESS_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
VAddr interval_end = std::min(address + size, vma->second.base + vma->second.size);
|
|
|
|
u32 interval_size = interval_end - interval_target;
|
|
|
|
u8* backing_memory = vma->second.backing_memory + (interval_target - vma->second.base);
|
|
|
|
backing_blocks.push_back({backing_memory, interval_size});
|
|
|
|
|
|
|
|
interval_target += interval_size;
|
|
|
|
}
|
|
|
|
return MakeResult(backing_blocks);
|
|
|
|
}
|
2017-11-07 11:35:17 -08:00
|
|
|
} // namespace Kernel
|