Skip to content
This repository has been archived by the owner on Aug 3, 2022. It is now read-only.

Latest commit

 

History

History

week-11

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 

Week 11

Overview

This week you will implement a SQLite benchmark program (SQLiteBench) using C/C++ SQLite API

Follow the guide below. If you have any questions, Please feel free to contact me via email (Jonghyeok Park / akindo19@skku.edu)

NOTE: This lab is based on the Linux environment. If you don't have a Linux machine, use VirturalBox. (Recommend Ubuntu 18.04)

Instructions

Prerequisite

  • git
sudo apt-get install git
  • cmake
sudo apt-get install cmake 
  • sqlite library
    1. go to the directory you have insatlled SQLite database in week 10.
    2. type sudo make install -j command (install library)
cd {PATH}/sqlite-src-3360000/build
make -j
sudo make install -j

0. Download SQLiteBench soruce code

git clone https://github.com/meeeejin/SWE3033-F2021.git
cd SWE3033-F2021/week-11

// if you already cloned the reposity then 
cd SWE3033-F2021
git pull

1. How to compile the SQLiteBench

cd SWE3033-F2021/week-11/sqlite3Bench
mkdir build && cd build
CC=gcc CXX=g++ cmake .. -DCMAKE_BUILD_TYPE=Release

2. How to run the SQLiteBench

./sqliteBench --journal_mode=off --benchmarks=directfillrandom

3. About SQLiteBench

  1. Input user option
  • SQLiteBench stores option given by the user in Benchmark in FLAGS_* variables
// main.cc
int main (int argc, char** argv) {

  init();
  char* default_db_path = static_cast<char*>(malloc(sizeof(char) * 1024));
  strcpy(default_db_path, "./");

  for (int i = 1; i < argc; i++) {
    double d;
    int n;
    char junk;
    char journal_mode[100];

    if (starts_with(argv[i], "--benchmarks=")) {
      FLAGS_benchmarks = argv[i] + strlen("--benchmarks=");
    } else if (sscanf(argv[i], "--histogram=%d%c", &n, &junk) == 1 &&
               (n == 0 || n == 1)) {
      FLAGS_histogram = n;
    } else if (sscanf(argv[i], "--raw=%d%c", &n, &junk) == 1 &&
               (n == 0 || n == 1)) {
      FLAGS_raw = n;
    } else if (sscanf(argv[i], "--compression_ratio=%lf%c", &d, &junk) == 1) {
      FLAGS_compression_ratio = d;
    } else if (sscanf(argv[i], "--use_existing_db=%d%c", &n, &junk) == 1 &&
               (n == 0 || n == 1)) {
      FLAGS_use_existing_db = n;
    }
    ... 
}
  1. Run Benchmark
  • Open database
// bench.h
class Benchmark {

 public:
  sqlite3* db_;
  int num_;
  // ellipsis

  void setJournalMode();
  void benchmark_open();

};

// bench.cc 
void Benchmark::benchmark_open() {
  assert(db_ == NULL);

  int status;
  char file_name[100];
  char* err_msg = NULL;
  db_num_++;

  /* Open database */
  char *tmp_dir = FLAGS_db;
  snprintf(file_name, sizeof(file_name),
            "%s/sqlitebench_test-%d.db",
            tmp_dir,
            db_num_);

  // (jhpark): always use fresh db
  if (!FLAGS_use_existing_db) {
    remove(file_name);
  }

  status = sqlite3_open(file_name, &db_);
  if (status) {
    fprintf(stderr, "open error: %s\n", sqlite3_errmsg(db_));
    exit(1);
  }
  • Run specific benchmark operation and measure the time for each operation
void Benchmark::benchmark_run() {
  print_header();
  benchmark_open();

  char* benchmarks = FLAGS_benchmarks;
  while (benchmarks != NULL) {
    char* sep = strchr(benchmarks, ',');
    char* name;
    if (sep == NULL) {
      name = benchmarks;
      benchmarks = NULL;
    } else {
      name = static_cast<char*>(calloc(sizeof(char), (sep - benchmarks + 1)));
      strncpy(name, benchmarks, sep - benchmarks);
      benchmarks = sep + 1;
    }   
    bytes_ = 0;
    
    // measure the time elapsed.
    start();
    bool known = true;
    bool write_sync = false;
    if (!strcmp(name, "fillseq")) {
      benchmark_write(write_sync, SEQUENTIAL, FRESH, num_, FLAGS_value_size, 1); 
      wal_checkpoint(db_);
    } // xxx(homework)
    else if (!strcmp(name, "directfillrandom")) {
      benchmark_directFillRand(num_);
    }
    ...

    if (known) {
      stop(name);
    }
  }
  1. Show Benchmark results
vldb@NVDIMM:~/SWE3033/cmake-project-template/build$ ./sqliteBench --journal_mode=wal --page_size=16384 --benchmarks=directfillrandom
journal_mode = wal
SQLite:     version 3.36.0
Date:       Sun Nov  7 22:50:27 2021
CPU:        16 * Intel(R) Xeon(R) CPU E5-2640 v3 @ 2.60GHz
CPUCache:   20480 KB
Keys:       16 bytes each
Values:     0 bytes each
Entries:    100000
RawSize:    1.5 MB (estimated)
------------------------------------------------
directfillrandom :      69.857 micros/op;

4. How to use SQLite API in SQLiteBench

  1. Link SQLite API in Cmake build configuration (CMakeList.txt)
project(sqlite_bench)

cmake_minimum_required(VERSION 3.19)

include(CheckCXXCompilerFlag)
CHECK_CXX_COMPILER_FLAG("-march=native" COMPILER_SUPPORTS_MARCH_NATIVE)
if(COMPILER_SUPPORTS_MARCH_NATIVE)
    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -march=native")
endif()

GET_FILENAME_COMPONENT(PROPATH src ABSOLUTE)
message(STATUS "source code dir: ${PROPATH}")

include_directories(include)
FILE(GLOB_RECURSE SRC_FILES ${PROPATH}/*.cc)
add_library(sqliteBenchLib ${SRC_FILES})
add_executable(sqliteBench main.cc)
target_link_libraries(sqliteBench sqliteBenchLib sqlite3)
  1. Set WAL jounral mode in SQLite databae engine using sqlite3_exec() function

[Warning] setWalJournalMode() function is sample for explanation, there is no setWalJournalMode() function in SQLiteBench program.

// sample code 
void Benchmark::setWalJournalMode() {
  int status;
  const char* WAL_stmt = "PRAGMA journal_mode = WAL";
  status = sqlite3_exec(db_, WAL_stmt, NULL, NULL, &err_msg);
}

5. Write Code (homewrok.cc)

  1. benchmark_setJournalMode()
  • Write a code for setting the journal mode in the SQLite database engine
  • [Requirement]
    • This function set jounral mode using "FLAGS_journal_mode" extern variable (user input)

    • This function checks journal mode; if user try to set wrong journal mode then return -2

    • This function returns status (int data type) of sqlite API function

        int Benchmark::benchmark_setJournalMode() {
          // fill this function
          return 0;
        }
  1. benchmark_setPageSize()
  • Write a code for setting page size in the SQLite database engine
  • [Requirement]
    • This function sets page size using "FLAGS_page_size" extern variable (user input)

    • This function returns status (int data type) of sqltie API function

    • This function is called at benchmark_open() in bench.cc

      int Benchmark::benchmark_setPageSize() {
       // please fill this function
       return 0;
      }
  1. benchmark_directFillRand(num_)
  • Write a code for insert function (direct SQL execution mode)

  • [Requriement]

    • This function fills random key-value data using direct qurey execution sqlite API function (please refer to lecture slide week 11)
    • This function has single arguemnt num_ which is total number of operations
    • This function creates SQL statement (key-value pair) do not modfiy this code
    • This function executes given SQL statement
    • This function is called at benchmark_open() in bench.cc
    int Benchmark::benchmark_directFillRand(int num_) {
     //      DO NOT MODIFY HERE     //
     const char* value = gen_.rand_gen_generate(FLAGS_value_size);
     char key[100];
     const int k = gen_.rand_next() % num_;
    
     snprintf(key, sizeof(key), "%016d", k); 
     char fill_stmt[100];
     snprintf(fill_stmt, sizeof(fill_stmt), "INSERT INTO test values (%s , %s)", key ,value);
     ////////////////////////////////
    
     // execute SQL statement
     // please fill this function
    
     return 0;
    }

Report Submission

  1. Follow the guideline (Insturuction #5) and write source code (SWE3033-F2021/week-11/sqlite3Bench/src/homework.cc)
  2. Write a simple comment (if any, less then 5 sentences) for your implementation in the source code and attach a screenshot of running the command below.
  • command
    ./sqliteBench --journal_mode=wal --page_size=16384 --benchmarks=directfillrandom
    
  • comment (example)
    • I used xxx API function for yyy and I used zzz to check the journal name

Organize the source code file (only SWE3033-F2021/week-11/sqlite3Bench/src/homework.cc file)and screenshot into the single zip file.

  • file name format: {Student ID}-{Name}-{week#}.pdf (e.g., 2021000000-JonghyeokPark-week11.zip)

Tip. How to build debug mode in CMake build configuration (Optional)

cd SWE3033-F2021/week-11/sqlite3Bench
mkdir build && cd debug_build
CC=gcc CXX=g++ cmake .. -DCMAKE_BUILD_TYPE=Debug

Reference