Skip to content

Commit e4af6ef

Browse files
committed
Add MemoryMapping source and test driver
1 parent 55dda45 commit e4af6ef

File tree

3 files changed

+317
-0
lines changed

3 files changed

+317
-0
lines changed

memory_mapping.cpp

+187
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,187 @@
1+
// -*- coding:utf-8; mode:c++; mode:auto-fill; fill-column:80; -*-
2+
3+
/// @file memory_mapping.cpp
4+
/// @brief Platform-independent memory mapping.
5+
/// @author J. Arrieta <[email protected]>
6+
/// @date January, 08, 2019
7+
/// @copyright (C) 2019 Nabla Zero Labs
8+
9+
// Related header
10+
#include "memory_mapping.hpp"
11+
12+
// C++ Standard Library
13+
#include <sstream>
14+
#include <stdexcept>
15+
#include <system_error>
16+
#include <utility>
17+
18+
// Platform-specific libraries.
19+
#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__))
20+
#include <fcntl.h>
21+
#include <stdio.h>
22+
#include <stdlib.h>
23+
#include <sys/mman.h>
24+
#include <sys/stat.h>
25+
#include <unistd.h>
26+
#elif defined(_WIN64)
27+
#include <Windows.h>
28+
#else
29+
#error "Your platform is not supported"
30+
#endif
31+
32+
namespace nzl {
33+
namespace {
34+
35+
#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__))
36+
auto make_mapping(const char* path) {
37+
auto fp = ::open(path, O_RDONLY);
38+
if (fp == -1) {
39+
throw std::system_error(errno, std::system_category(), path);
40+
}
41+
42+
struct stat sb;
43+
if (::fstat(fp, &sb) == -1) {
44+
::close(fp);
45+
throw std::system_error(errno, std::system_category(), path);
46+
}
47+
48+
auto data = ::mmap(nullptr, sb.st_size, PROT_READ, MAP_SHARED, fp, 0);
49+
::close(fp); // closing file descriptor does not unmap region
50+
51+
if (data == MAP_FAILED) {
52+
throw std::system_error(errno, std::system_category(), path);
53+
}
54+
55+
return std::make_pair(static_cast<std::byte*>(data),
56+
static_cast<std::size_t>(sb.st_size));
57+
}
58+
59+
auto clear_mapping(std::byte* data, std::size_t size) noexcept {
60+
if (data != nullptr) {
61+
::munmap(static_cast<void*>(data), size);
62+
data = nullptr;
63+
size = 0;
64+
}
65+
return std::make_pair(data, size);
66+
}
67+
68+
#elif defined(_WIN64)
69+
// Get the last Windows error as a C++ std::string.
70+
std::string last_error() {
71+
DWORD errorMessageID = ::GetLastError();
72+
if (errorMessageID == 0) {
73+
return {};
74+
} else {
75+
LPSTR messageBuffer = nullptr;
76+
size_t size = FormatMessageA(
77+
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |
78+
FORMAT_MESSAGE_IGNORE_INSERTS,
79+
nullptr, errorMessageID, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
80+
(LPSTR)&messageBuffer, 0, nullptr);
81+
std::string message(messageBuffer, size);
82+
LocalFree(messageBuffer);
83+
return message;
84+
}
85+
}
86+
87+
auto make_mapping(const char* path) {
88+
HANDLE hfile = ::CreateFileA(path, GENERIC_READ, FILE_SHARE_READ, nullptr,
89+
OPEN_EXISTING, 0, nullptr);
90+
if (hfile == INVALID_HANDLE_VALUE) {
91+
std::ostringstream oss;
92+
oss << "error opening \"" << path << "\": " << last_error();
93+
throw std::runtime_error(oss.str());
94+
}
95+
96+
LARGE_INTEGER size_value;
97+
if (!::GetFileSizeEx(hfile, &size_value)) {
98+
std::ostringstream oss;
99+
oss << "cannot obtain size of \"" << path << "\": " << last_error();
100+
throw std::runtime_error(oss.str());
101+
}
102+
auto size = size_value.QuadPart;
103+
104+
HANDLE hmap =
105+
::CreateFileMapping(hfile, nullptr, PAGE_READONLY, 0, 0, nullptr);
106+
// closing the file handle does not affect the mapping
107+
if (!::CloseHandle(hfile) || (hmap == nullptr)) {
108+
std::ostringstream oss;
109+
oss << "error creating file mapping of \"" << path
110+
<< "\": " << last_error();
111+
throw std::runtime_error(oss.str());
112+
}
113+
114+
auto data = ::MapViewOfFile(hmap, FILE_MAP_READ, 0, 0, size);
115+
// closing the map handle does not unmap the region
116+
if (!::CloseHandle(hmap) || (data == nullptr)) {
117+
std::ostringstream oss;
118+
oss << "error finalizing creation of file mapping of \"" << path
119+
<< "\": " << last_error();
120+
throw std::runtime_error(oss.str());
121+
}
122+
123+
return std::make_pair(static_cast<std::byte*>(data),
124+
static_cast<std::size_t>(size));
125+
}
126+
127+
auto clear_mapping(std::byte* data, std::size_t size) noexcept {
128+
if (data != nullptr) {
129+
::UnmapViewOfFile(data);
130+
data = nullptr;
131+
size = 0;
132+
}
133+
return std::make_pair(data, size);
134+
}
135+
#else
136+
#error "Your platform is not supported"
137+
#endif
138+
139+
} // anonymous namespace
140+
141+
MemoryMapping::MemoryMapping(const std::string& path) {
142+
std::tie(m_data, m_size) = make_mapping(path.c_str());
143+
}
144+
145+
MemoryMapping::MemoryMapping(MemoryMapping&& other) noexcept
146+
: m_data{other.m_data}, m_size{other.m_size} {
147+
other.m_data = nullptr;
148+
other.m_size = 0;
149+
}
150+
151+
MemoryMapping& MemoryMapping::operator=(MemoryMapping&& other) noexcept {
152+
m_data = other.m_data;
153+
m_size = other.m_size;
154+
155+
other.m_data = nullptr;
156+
other.m_size = 0;
157+
158+
return *this;
159+
}
160+
161+
MemoryMapping::~MemoryMapping() noexcept {
162+
std::tie(m_data, m_size) = clear_mapping(m_data, m_size);
163+
}
164+
165+
const std::byte* MemoryMapping::data() const noexcept { return m_data; }
166+
167+
const std::byte* MemoryMapping::begin() const noexcept { return cbegin(); }
168+
169+
const std::byte* MemoryMapping::end() const noexcept { return cend(); }
170+
171+
const std::byte* MemoryMapping::cbegin() const noexcept { return m_data; }
172+
173+
const std::byte* MemoryMapping::cend() const noexcept {
174+
return std::next(cbegin(), size());
175+
}
176+
177+
std::size_t MemoryMapping::size() const noexcept { return m_size; }
178+
179+
void MemoryMapping::swap(MemoryMapping& other) noexcept {
180+
using std::swap;
181+
swap(m_data, other.m_data);
182+
swap(m_size, other.m_size);
183+
}
184+
185+
void swap(MemoryMapping& lhs, MemoryMapping& rhs) noexcept { lhs.swap(rhs); }
186+
187+
} // namespace nzl

memory_mapping.hpp

+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
// -*- coding:utf-8; mode:c++; mode:auto-fill; fill-column:80; -*-
2+
3+
/// @file memory_mapping.hpp
4+
/// @brief Platform-independent memory mapping.
5+
/// @author J. Arrieta <[email protected]>
6+
/// @date January, 08, 2019
7+
/// @copyright (C) 2019 Nabla Zero Labs
8+
9+
#pragma once
10+
11+
// C++ Standard Library
12+
#include <cstdint>
13+
#include <string>
14+
15+
namespace nzl {
16+
17+
class MemoryMapping {
18+
public:
19+
/// @brief Create a read-only MemoryMaping of file.
20+
/// @param path Filesystem path to the file.
21+
/// @throws std::system_error (Linux/macOS) if mapping cannot be created.
22+
/// @throws std::runtime_error (Windows) if mapping cannot be created.
23+
MemoryMapping(const std::string& path);
24+
25+
/// @brief Move constructor.
26+
MemoryMapping(MemoryMapping&& other) noexcept;
27+
28+
/// @brief Move assignment.
29+
MemoryMapping& operator=(MemoryMapping&& other) noexcept;
30+
31+
/// @brief Destructor.
32+
~MemoryMapping() noexcept;
33+
34+
/// @brief Return a pointer to the beginning of the data block.
35+
const std::byte* data() const noexcept;
36+
37+
/// @brief Return a pointer to the beginning of the data block.
38+
const std::byte* begin() const noexcept;
39+
40+
/// @brief Return a pointer to the end of the data block.
41+
const std::byte* end() const noexcept;
42+
43+
/// @brief Return a pointer to the beginning
44+
const std::byte* cbegin() const noexcept;
45+
46+
/// @brief Return a pointer to the end of the data block.
47+
const std::byte* cend() const noexcept;
48+
49+
/// @brief Return the size of the mapped region in bytes.
50+
std::size_t size() const noexcept;
51+
52+
/// @brief Swap this and other MemoryMapping
53+
/// @param other The MemoryMapping to swap with this one.
54+
void swap(MemoryMapping& other) noexcept;
55+
56+
private:
57+
std::byte* m_data{nullptr};
58+
std::size_t m_size{0};
59+
};
60+
61+
/// @brief Swap two @link MemoryMapping MemoryMappings@endlink
62+
/// @param lhs First MemoryMapping to swap.
63+
/// @param rhs Second MemoryMapping to swap.
64+
void swap(MemoryMapping& lhs, MemoryMapping& rhs) noexcept;
65+
66+
} // namespace nzl

memory_mapping.t.cpp

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
// -*- coding:utf-8; mode:c++; mode:auto-fill; fill-column:80; -*-
2+
3+
/// @file memory_mapping.t.cpp
4+
/// @brief Test driver for platform-independent memory mapping.
5+
/// @author J. Arrieta <[email protected]>
6+
/// @date January, 08, 2019
7+
/// @copyright (C) 2019 Nabla Zero Labs
8+
9+
// Nabla Zero Labs Library.
10+
#include "memory_mapping.hpp"
11+
12+
// C++ Standard Library
13+
#include <algorithm>
14+
#include <chrono>
15+
#include <cstdlib>
16+
#include <fstream>
17+
#include <iomanip>
18+
#include <iostream>
19+
#include <iterator>
20+
21+
void using_file(const std::string& path) {
22+
std::ifstream fp(path, std::ios::in | std::ios::binary);
23+
24+
std::istreambuf_iterator<char> it(fp);
25+
const std::istreambuf_iterator<char> it_end{};
26+
27+
std::size_t n{0};
28+
while (it != it_end) {
29+
if (*it++ == '\n') ++n;
30+
}
31+
std::cout << "[f] " << std::setw(8) << n << " " << path << "\n";
32+
}
33+
34+
void using_mapping(const std::string& path) {
35+
nzl::MemoryMapping mm{path};
36+
37+
std::size_t n{0};
38+
for (auto&& c : mm) {
39+
if (c == std::byte('\n')) ++n;
40+
}
41+
std::cout << "[m] " << std::setw(8) << n << " " << path << "\n";
42+
}
43+
44+
int main(int argc, char* argv[]) {
45+
using namespace std::chrono;
46+
47+
bool use_map = std::getenv("NZL_USE_MAP") != nullptr;
48+
if (use_map) {
49+
std::cout << "using memory mapping\n";
50+
auto cpu_beg = system_clock::now();
51+
std::for_each(argv + 1, argv + argc, using_mapping);
52+
auto cpu_end = system_clock::now();
53+
auto cpu_dur = nanoseconds(cpu_end - cpu_beg).count();
54+
std::cout << cpu_dur << " nanoseconds elapsed\n";
55+
56+
} else {
57+
std::cout << "using file\n";
58+
auto cpu_beg = system_clock::now();
59+
std::for_each(argv + 1, argv + argc, using_file);
60+
auto cpu_end = system_clock::now();
61+
auto cpu_dur = nanoseconds(cpu_end - cpu_beg).count();
62+
std::cout << cpu_dur << " nanoseconds elapsed\n";
63+
}
64+
}

0 commit comments

Comments
 (0)