summaryrefslogtreecommitdiffstats
path: root/src/video_core/buffer_cache/map_interval.h
blob: ef974b08ae8075c4147904d2d2411e06041f2dc6 (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
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
// Copyright 2019 yuzu Emulator Project
// Licensed under GPLv2 or any later version
// Refer to the license.txt file included.

#pragma once

#include <array>
#include <cstddef>
#include <memory>
#include <vector>

#include <boost/intrusive/set_hook.hpp>

#include "common/common_types.h"
#include "video_core/gpu.h"

namespace VideoCommon {

struct MapInterval : public boost::intrusive::set_base_hook<boost::intrusive::optimize_size<true>> {
    MapInterval() = default;

    /*implicit*/ MapInterval(VAddr start_) noexcept : start{start_} {}

    explicit MapInterval(VAddr start_, VAddr end_, GPUVAddr gpu_addr_) noexcept
        : start{start_}, end{end_}, gpu_addr{gpu_addr_} {}

    bool IsInside(VAddr other_start, VAddr other_end) const noexcept {
        return start <= other_start && other_end <= end;
    }

    bool Overlaps(VAddr other_start, VAddr other_end) const noexcept {
        return start < other_end && other_start < end;
    }

    void MarkAsModified(bool is_modified_, u64 ticks_) noexcept {
        is_modified = is_modified_;
        ticks = ticks_;
    }

    boost::intrusive::set_member_hook<> member_hook_;
    VAddr start = 0;
    VAddr end = 0;
    GPUVAddr gpu_addr = 0;
    u64 ticks = 0;
    bool is_written = false;
    bool is_modified = false;
    bool is_registered = false;
    bool is_memory_marked = false;
    bool is_sync_pending = false;
};

struct MapIntervalCompare {
    constexpr bool operator()(const MapInterval& lhs, const MapInterval& rhs) const noexcept {
        return lhs.start < rhs.start;
    }
};

class MapIntervalAllocator {
public:
    MapIntervalAllocator();
    ~MapIntervalAllocator();

    MapInterval* Allocate() {
        if (free_list.empty()) {
            AllocateNewChunk();
        }
        MapInterval* const interval = free_list.back();
        free_list.pop_back();
        return interval;
    }

    void Release(MapInterval* interval) {
        free_list.push_back(interval);
    }

private:
    struct Chunk {
        std::unique_ptr<Chunk> next;
        std::array<MapInterval, 0x8000> data;
    };

    void AllocateNewChunk();

    void FillFreeList(Chunk& chunk);

    std::vector<MapInterval*> free_list;

    Chunk first_chunk;

    std::unique_ptr<Chunk>* new_chunk = &first_chunk.next;
};

} // namespace VideoCommon