-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathring_buffer.cpp
More file actions
99 lines (92 loc) · 3.03 KB
/
ring_buffer.cpp
File metadata and controls
99 lines (92 loc) · 3.03 KB
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
94
95
96
97
98
99
#include "ring_buffer.h"
#include <unistd.h>
#include <sys/mman.h>
#include <atomic>
#include <cstring>
#include <iostream>
#include <errno.h>
namespace event_engine
{
RingBuffer::RingBuffer()
{
}
RingBuffer::RingBuffer(int fd, uint32_t pages) : fd_{fd}, pages_(pages)
{
}
void RingBuffer::SetTimeOffset(uint64_t offset)
{
timeOffset_ = offset;
}
bool RingBuffer::Mmap(std::string &err)
{
if (pages_ & (pages_ - 1) != 0) // pages not pow of 2
{
err = "pages not pow of 2";
return false;
}
int page_size = getpagesize();
size_ = page_size * (pages_ + 1);
data_size_ = page_size * pages_;
data_mask_ = data_size_ - 1;
void *maped = mmap(nullptr, size_, PROT_WRITE | PROT_READ, MAP_SHARED, fd_, 0);
if (maped == MAP_FAILED)
{
err = strerror(errno);
return false;
}
data_ = (char *)maped + page_size;
meta_ = (perf_event_mmap_page *)maped;
return true;
}
void RingBuffer::Unmap()
{
munmap((void *)meta_, size_);
}
uint64_t RingBuffer::TimeRunning()
{
return std::atomic_load_explicit((std::atomic<uint64_t> *)&meta_->time_running, std::memory_order_relaxed);
}
std::vector<std::pair<int, char *>> RingBuffer::Read()
{
if (data_ == nullptr)
{
return std::vector<std::pair<int, char *>>();
}
int data_tail = meta_->data_tail;
int data_head = std::atomic_load_explicit((std::atomic<uint64_t> *)&meta_->data_head, std::memory_order_acquire);
int data_begin, data_end;
std::vector<std::pair<int, char *>> res; // free manual
while (data_tail != data_head)
{
data_begin = data_tail & data_mask_;
data_end = data_head & data_mask_;
if (data_end > data_begin)
{
char *buff = new char[data_end - data_begin];
if (buff != nullptr)
{
std::memcpy(buff, data_ + data_begin, data_end - data_begin);
res.push_back(std::pair<int, char *>{data_end - data_begin, buff});
}
}
else
{
char *buff = new char[data_size_ - data_begin + data_end];
if (buff != nullptr)
{
std::memcpy(buff, data_ + data_begin, data_size_ - data_begin);
std::memcpy(buff, data_, data_end);
res.push_back(std::pair<int, char *>{data_size_ - data_begin + data_end, buff});
}
}
data_tail = data_head;
std::atomic_store_explicit((std::atomic<uint64_t> *)&meta_->data_tail, data_tail, std::memory_order_release);
data_head = std::atomic_load_explicit((std::atomic<uint64_t> *)&meta_->data_head, std::memory_order_acquire);
}
return res;
}
uint64_t RingBuffer::GetTimeOffset()
{
return timeOffset_;
}
}