Skip to content

Commit

Permalink
Fix Circular Queue Destructor + Memory Infos + Improve Priority Docum…
Browse files Browse the repository at this point in the history
…entation
  • Loading branch information
PixelyIon committed Nov 22, 2020
1 parent 7167393 commit a3dd759
Show file tree
Hide file tree
Showing 7 changed files with 34 additions and 33 deletions.
17 changes: 6 additions & 11 deletions app/src/main/cpp/skyline/common/circular_queue.h
Original file line number Diff line number Diff line change
Expand Up @@ -21,19 +21,14 @@ namespace skyline {
std::condition_variable produceCondition;

public:
inline CircularQueue(size_t size) : vector(size * sizeof(Type)) {}
inline CircularQueue(size_t size) : vector((size + 1) * sizeof(Type)) {}

inline ~CircularQueue() {
ssize_t size{};
if (start < end)
size = end - start;
else
size = (reinterpret_cast<Type *>(vector.end().base()) - start) + (end - reinterpret_cast<Type *>(vector.begin().base()));

while (size--) {
std::destroy_at(start);
if (start + 1 == reinterpret_cast<Type *>(vector.end().base()))
start = reinterpret_cast<Type *>(vector.begin().base());
while (start != end) {
auto next{start + 1};
next = (next == reinterpret_cast<Type *>(vector.end().base())) ? reinterpret_cast<Type *>(vector.begin().base()) : next;
std::destroy_at(next);
start = next;
}
}

Expand Down
10 changes: 5 additions & 5 deletions app/src/main/cpp/skyline/kernel/memory.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -195,18 +195,18 @@ namespace skyline::kernel {
return std::nullopt;
}

size_t MemoryManager::GetMemoryUsage() {
size_t MemoryManager::GetUserMemoryUsage() {
std::shared_lock lock(mutex);
size_t size{};
for (const auto &chunk : chunks)
if (chunk.state != memory::states::Unmapped && chunk.state != memory::states::Reserved)
if (chunk.state == memory::states::Heap)
size += chunk.size;
return size;
return size + code.size + state.process->mainThreadStack->size;
}

size_t MemoryManager::GetKMemoryBlockSize() {
size_t MemoryManager::GetSystemResourceUsage() {
std::shared_lock lock(mutex);
constexpr size_t KMemoryBlockSize{0x40};
return util::AlignUp(chunks.size() * KMemoryBlockSize, PAGE_SIZE);
return std::min(static_cast<size_t>(state.process->npdm.meta.systemResourceSize), util::AlignUp(chunks.size() * KMemoryBlockSize, PAGE_SIZE));
}
}
7 changes: 4 additions & 3 deletions app/src/main/cpp/skyline/kernel/memory.h
Original file line number Diff line number Diff line change
Expand Up @@ -240,14 +240,15 @@ namespace skyline {
std::optional<ChunkDescriptor> Get(void *ptr);

/**
* @return The cumulative size of all memory mappings in bytes
* @return The cumulative size of all heap (Physical Memory + Process Heap) memory mappings, the code region and the main thread stack in bytes
*/
size_t GetMemoryUsage();
size_t GetUserMemoryUsage();

/**
* @return The total page-aligned size used to store memory block metadata, if they were KMemoryBlocks rather than ChunkDescriptor
* @note There is a ceiling of SystemResourceSize as specified in the NPDM, this value will be clipped to that
*/
size_t GetKMemoryBlockSize();
size_t GetSystemResourceUsage();
};
}
}
16 changes: 10 additions & 6 deletions app/src/main/cpp/skyline/kernel/scheduler.h
Original file line number Diff line number Diff line change
Expand Up @@ -13,15 +13,19 @@ namespace skyline {
namespace kernel {
using CoreMask = std::bitset<constant::CoreCount>;

/**
* @brief Priority on HOS determines scheduling behavior relative to other threads
* @note Lower priority values result in a higher priority, similar to niceness on Linux
*/
struct Priority {
u8 min;
u8 max;
u8 min; //!< Numerically lowest priority, highest scheduler priority
u8 max; //!< Numerically highest priority, lowest scheduler priority

/**
* @return A bitmask with each bit corresponding to if scheduler priority with the same index is valid
*/
constexpr u64 Mask() const {
u64 mask{};
for (u8 i{min}; i <= max; i++)
mask |= 1 << i;
return mask;
return (std::numeric_limits<u64>::max() >> ((std::numeric_limits<u64>::digits - 1 + min) - max)) << min;
}

constexpr bool Valid(i8 value) const {
Expand Down
12 changes: 6 additions & 6 deletions app/src/main/cpp/skyline/kernel/svc.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -367,7 +367,7 @@ namespace skyline::kernel::svc {
}

auto processMask{state.process->npdm.threadInfo.coreMask};
if ((processMask | affinityMask) == processMask) {
if ((processMask | affinityMask) != processMask) {
state.logger->Warn("svcSetThreadCoreMask: 'affinityMask' invalid: {} (Process Mask: {})", affinityMask, processMask);
state.ctx->gpr.w0 = result::InvalidCoreId;
return;
Expand Down Expand Up @@ -790,11 +790,11 @@ namespace skyline::kernel::svc {
break;

case InfoState::TotalMemoryAvailable:
out = totalPhysicalMemory;
out = std::min(totalPhysicalMemory, state.process->memory.heap.size);
break;

case InfoState::TotalMemoryUsage:
out = state.process->memory.GetMemoryUsage() + state.process->memory.GetKMemoryBlockSize();
out = state.process->memory.GetUserMemoryUsage() + state.process->memory.GetSystemResourceUsage();
break;

case InfoState::RandomEntropy:
Expand Down Expand Up @@ -823,19 +823,19 @@ namespace skyline::kernel::svc {

case InfoState::TotalSystemResourceUsage:
// A very rough approximation of what this should be on the Switch, the amount of memory allocated for storing the memory blocks (https://switchbrew.org/wiki/Kernel_objects#KMemoryBlockManager)
out = std::min(static_cast<size_t>(state.process->npdm.meta.systemResourceSize), state.process->memory.GetKMemoryBlockSize());
out = state.process->memory.GetSystemResourceUsage();
break;

case InfoState::ProgramId:
out = state.process->npdm.aci0.programId;
break;

case InfoState::TotalMemoryAvailableWithoutSystemResource:
out = totalPhysicalMemory - state.process->npdm.meta.systemResourceSize;
out = std::min(totalPhysicalMemory, state.process->memory.heap.size) - state.process->npdm.meta.systemResourceSize;
break;

case InfoState::TotalMemoryUsageWithoutSystemResource:
out = state.process->memory.GetMemoryUsage(); // Our regular estimates don't contain the system resources
out = state.process->memory.GetUserMemoryUsage();
break;

case InfoState::UserExceptionContextAddr:
Expand Down
2 changes: 1 addition & 1 deletion app/src/main/cpp/skyline/vfs/npdm.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ namespace skyline::vfs {
auto trailingOnes{__builtin_ctz(~capability.raw)};
switch (trailingOnes) {
case 3:
threadInfo.priority = kernel::Priority{capability.threadInfo.lowestPriority, capability.threadInfo.highestPriority};
threadInfo.priority = kernel::Priority{capability.threadInfo.highestPriority, capability.threadInfo.lowestPriority};

threadInfo.coreMask = {};
for (u8 core{capability.threadInfo.minCoreId}; core <= capability.threadInfo.maxCoreId; core++)
Expand Down
3 changes: 2 additions & 1 deletion app/src/main/cpp/skyline/vfs/npdm.h
Original file line number Diff line number Diff line change
Expand Up @@ -81,11 +81,12 @@ namespace skyline {
union __attribute__((packed)) NpdmKernelCapability {
/**
* @url https://switchbrew.org/wiki/NPDM#ThreadInfo
* @note Priority field names are based on real scheduler priority (Lower value is higher priority)
*/
struct __attribute__((packed)) {
u8 pattern : 4; //!< 0b0111
u8 highestPriority : 6;
u8 lowestPriority : 6;
u8 highestPriority : 6;
u8 minCoreId : 8;
u8 maxCoreId : 8;
} threadInfo;
Expand Down

0 comments on commit a3dd759

Please sign in to comment.