#include <iostream>
#include <chrono>
#include <numeric>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <linux/fs.h>
#include <unistd.h>
#include <sys/swap.h>
using namespace std;
const char zram_blkdev_path[] = "/dev/block/zram0";
const size_t sector_size = 512;
const size_t page_size = 4096;
void fillPageRand(uint32_t *page) {
int start = rand();
for (int i = 0; i < page_size / sizeof(int); i++) {
page[i] = start+i;
}
}
void fillPageCompressible(uint32_t *page) {
int val = rand() & 0xfff;
for (int i = 0; i < page_size / sizeof(int); i++) {
page[i] = val;
}
}
class AlignedAlloc {
void *m_ptr;
public:
AlignedAlloc(size_t size, size_t align) {
posix_memalign(&m_ptr, align, size);
}
~AlignedAlloc() {
free(m_ptr);
}
void *ptr() {
return m_ptr;
}
};
class BlockFd {
int m_fd = -1;
public:
BlockFd(const char *path, bool direct) {
m_fd = open(path, O_RDWR | (direct ? O_DIRECT : 0));
}
size_t getSize() {
size_t blockSize = 0;
int result = ioctl(m_fd, BLKGETSIZE, &blockSize);
if (result < 0) {
cout << "ioctl failed" << endl;
}
return blockSize * sector_size;
}
~BlockFd() {
if (m_fd >= 0) {
close(m_fd);
}
}
void fillWithCompressible() {
size_t devSize = getSize();
AlignedAlloc page(page_size, page_size);
for (uint64_t offset = 0; offset < devSize; offset += page_size) {
fillPageCompressible((uint32_t*)page.ptr());
ssize_t ret = write(m_fd, page.ptr(), page_size);
if (ret != page_size) {
cout << "write() failed" << endl;
}
}
}
void benchSequentialRead() {
chrono::time_point<chrono::high_resolution_clock> start, end;
size_t devSize = getSize();
size_t passes = 4;
AlignedAlloc page(page_size, page_size);
start = chrono::high_resolution_clock::now();
for (int i = 0; i < passes; i++) {
for (uint64_t offset = 0; offset < devSize; offset += page_size) {
if (offset == 0)
lseek(m_fd, offset, SEEK_SET);
ssize_t ret = read(m_fd, page.ptr(), page_size);
if (ret != page_size) {
cout << "read() failed" << endl;
}
}
}
end = chrono::high_resolution_clock::now();
size_t duration = chrono::duration_cast<chrono::microseconds>(end - start).count();
cout << "read: " << (double)devSize * passes / 1024.0 / 1024.0 / (duration / 1000.0 / 1000.0) << "MB/s" << endl;
}
void benchSequentialWrite() {
chrono::time_point<chrono::high_resolution_clock> start, end;
size_t devSize = getSize();
size_t passes = 4;
AlignedAlloc page(page_size, page_size);
start = chrono::high_resolution_clock::now();
for (int i = 0; i < passes; i++) {
for (uint64_t offset = 0; offset < devSize; offset += page_size) {
fillPageCompressible((uint32_t*)page.ptr());
if (offset == 0)
lseek(m_fd, offset, SEEK_SET);
ssize_t ret = write(m_fd, page.ptr(), page_size);
if (ret != page_size) {
cout << "write() failed" << endl;
}
}
}
end = chrono::high_resolution_clock::now();
size_t duration = chrono::duration_cast<chrono::microseconds>(end - start).count();
cout << "write: " << (double)devSize * passes / 1024.0 / 1024.0 / (duration / 1000.0 / 1000.0) << "MB/s" << endl;
}
};
int bench(bool direct)
{
BlockFd zramDev{zram_blkdev_path, direct};
zramDev.fillWithCompressible();
zramDev.benchSequentialRead();
zramDev.benchSequentialWrite();
return 0;
}
int main(int argc, char *argv[])
{
int result = swapoff(zram_blkdev_path);
if (result < 0) {
cout << "swapoff failed: " << strerror(errno) << endl;
}
bench(1);
result = system((string("mkswap ") + string(zram_blkdev_path)).c_str());
if (result < 0) {
cout << "mkswap failed: " << strerror(errno) << endl;
return -1;
}
result = swapon(zram_blkdev_path, 0);
if (result < 0) {
cout << "swapon failed: " << strerror(errno) << endl;
return -1;
}
return 0;
}