summaryrefslogtreecommitdiffstats
path: root/src/core/hle/kernel/resource_limit.cpp
blob: d9beaa3a40c2b147937ce5b1c98bc79588c0768a (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
// Copyright 2015 Citra Emulator Project
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.

#include "core/hle/kernel/errors.h"
#include "core/hle/kernel/resource_limit.h"
#include "core/hle/result.h"

namespace Kernel {
namespace {
constexpr std::size_t ResourceTypeToIndex(ResourceType type) {
    return static_cast<std::size_t>(type);
}
} // Anonymous namespace

ResourceLimit::ResourceLimit(KernelCore& kernel) : Object{kernel} {}
ResourceLimit::~ResourceLimit() = default;

bool ResourceLimit::Reserve(ResourceType resource, s64 amount) {
    return Reserve(resource, amount, 10000000000);
}

bool ResourceLimit::Reserve(ResourceType resource, s64 amount, u64 timeout) {
    const std::size_t index{ResourceTypeToIndex(resource)};

    s64 new_value = current[index] + amount;
    while (new_value > limit[index] && available[index] + amount <= limit[index]) {
        // TODO(bunnei): This is wrong for multicore, we should wait the calling thread for timeout
        new_value = current[index] + amount;

        if (timeout >= 0) {
            break;
        }
    }

    if (new_value <= limit[index]) {
        current[index] = new_value;
        return true;
    }
    return false;
}

void ResourceLimit::Release(ResourceType resource, u64 amount) {
    Release(resource, amount, amount);
}

void ResourceLimit::Release(ResourceType resource, u64 used_amount, u64 available_amount) {
    const std::size_t index{ResourceTypeToIndex(resource)};

    current[index] -= used_amount;
    available[index] -= available_amount;
}

std::shared_ptr<ResourceLimit> ResourceLimit::Create(KernelCore& kernel) {
    return std::make_shared<ResourceLimit>(kernel);
}

s64 ResourceLimit::GetCurrentResourceValue(ResourceType resource) const {
    return limit.at(ResourceTypeToIndex(resource)) - current.at(ResourceTypeToIndex(resource));
}

s64 ResourceLimit::GetMaxResourceValue(ResourceType resource) const {
    return limit.at(ResourceTypeToIndex(resource));
}

ResultCode ResourceLimit::SetLimitValue(ResourceType resource, s64 value) {
    const std::size_t index{ResourceTypeToIndex(resource)};
    if (current[index] <= value) {
        limit[index] = value;
        return RESULT_SUCCESS;
    } else {
        LOG_ERROR(Kernel, "Limit value is too large! resource={}, value={}, index={}",
                  static_cast<u32>(resource), value, index);
        return ERR_INVALID_STATE;
    }
}
} // namespace Kernel