当前位置:网站首页>The establishment of simple data cache layer
The establishment of simple data cache layer
2022-08-04 19:34:00 【Hong Dayu】
#include <cstddef>
#include <cstdint>
#include <cstdlib>
#include <cstring>
#include <fstream>
#include <iostream>
#include <map>
#include <sqlite3.h>
#include <string>
#include <utility>
#include <vector>
// Web Or Desktop App also need sqlite3 info
// ID,name,mode,var_type,coefficient,len,V0,min integer,max integer,pro integer
// virtual data layer operator sqlite3
#define ID 0x0
#define NAME 0x1
#define MODE 0x2
#define TYPE 0x3
#define COEF 0x4
#define LEN 0x5
#define VALUE 0x6
#define MAX 0x7
#define MIN 0x8
#define ATTR 0x9
#define SQL_COL 10
#define sqlite3_table "rau"
#define SQLITE_FIRST_CREAT 0x4
#define SQLITE_TABLE_OK 0x2
#define SQLITE_TABLE_NOT_OK 0x8
static const char create_table_sql[] = "CREATE TABLE " sqlite3_table "("
"ID INT PRIMARY KEY NOT NULL,"
"NAME TEXT NOT NULL,"
"MODE TEXT NOT NULL,"
"TYPE TEXT NOT NULL,"
"COEF INT NOT NULL,"
"LEN INT NOT NULL,"
"VALUE TEXT NOT NULL,"
"MAX INT NOT NULL,"
"MIN INT NOT NULL,"
"ATTR INT NOT NULL);";
static const char check_table_sql[] = "SELECT count(*) FROM sqlite_master"
" WHERE type='table'"
" AND name = \"" sqlite3_table "\"";
struct sqlite_row_data {
union {
uint32_t u32;
uint16_t u16;
uint8_t u8;
int32_t i32;
int16_t i16;
int8_t i8;
uint8_t bit : 1;
char str[256];
} val; // user programe function use
uint32_t id;
std::string name;
std::string mode;
std::string type;
std::string _val; // use sql to insert
uint32_t len;
uint32_t coeff;
uint32_t min;
uint32_t max;
uint32_t attr;
struct {
int32_t (*exec)(void *);
void *args;
} event_binds;
};
static std::map<uint32_t, sqlite_row_data> virtual_sqlite_layer;
class sqlite3_instance {
private:
uint32_t table_ok = 0x0;
sqlite3 *database = nullptr;
const char *db_path = nullptr;
public:
explicit sqlite3_instance(const char *path);
int32_t connect();
int32_t init_virtual_sqlite(const char *config,
std::map<uint32_t, sqlite_row_data> &data);
int32_t add(uint32_t index, sqlite_row_data val);
int32_t modify(uint32_t index, sqlite_row_data val);
int32_t querry(uint32_t index, sqlite_row_data &val);
int32_t del(uint32_t index);
int32_t sql_exec(const char *sql);
~sqlite3_instance();
};
/** * @brief FIXME: add mutex * */
sqlite3_instance::sqlite3_instance(const char *path) {
if (path == nullptr) {
throw "ERROR:sqlite3 need database file path";
}
this->db_path = path;
}
int32_t sqlite3_instance::connect() {
sqlite3_stmt *stmt;
int32_t ret = sqlite3_open(this->db_path, &(this->database));
if (ret != SQLITE_OK) {
printf("WARN:%s\n", sqlite3_errstr(ret));
} else {
ret =
sqlite3_prepare_v2(this->database, check_table_sql, -1, &stmt, nullptr);
if (ret != SQLITE_OK) {
printf("Sqlite3Warn:%s\n", sqlite3_errstr(ret));
} else {
if (sqlite3_step(stmt) == SQLITE_ROW) {
table_ok = (uint32_t)sqlite3_column_int(stmt, 0);
}
if (!table_ok) {
ret = sqlite3_exec(this->database, create_table_sql, nullptr, nullptr,
nullptr);
if (ret != SQLITE_OK) {
printf("Sqlite3Error:%s\n", sqlite3_errstr(ret));
}
table_ok |= SQLITE_FIRST_CREAT; // FIXME: first create table;
} else {
table_ok |= SQLITE_TABLE_OK;
}
}
}
return 0;
}
int32_t sqlite3_instance::sql_exec(const char *sql) {
int32_t ret = sqlite3_exec(this->database, sql, nullptr, nullptr, nullptr);
if (ret != SQLITE_OK) {
printf("Sqlite3Error:%s\n", sqlite3_errstr(ret));
return ret;
}
return 0;
}
int32_t sqlite3_instance::del(uint32_t index) {
const char sql[] = "DELETE FROM " sqlite3_table " WHERE ID=";
std::string _sql = std::string(sql) + std::to_string(index);
int ret =
sqlite3_exec(this->database, _sql.c_str(), nullptr, nullptr, nullptr);
if (ret != SQLITE_OK) {
printf("WARN:%s\n", sqlite3_errstr(ret));
return ret;
}
return 0;
}
// string 可以完成值的赋值和拷贝
/** * @brief struct filed with struct sqlite3_stmt * * @param val * @param stmt */
void struct_sqlite_row(sqlite_row_data &val, sqlite3_stmt *stmt) {
val.id = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, ID),
nullptr, 16);
val.name = std::string((const char *)sqlite3_column_text(stmt, NAME));
val.type = std::string((const char *)sqlite3_column_text(stmt, TYPE));
val.attr = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, ATTR),
nullptr, 16);
val.coeff = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, COEF),
nullptr, 16);
val.len = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, LEN),
nullptr, 16);
val.mode = std::string((const char *)sqlite3_column_text(stmt, MODE));
val._val = std::string((const char *)sqlite3_column_text(stmt, VALUE));
val.max = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, MAX),
nullptr, 16);
val.min = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, MIN),
nullptr, 16);
}
/** * @brief only one result for sqlite3 querry * * @tparam T * @param index * @param value * @return int32_t */
int32_t sqlite3_instance::querry(uint32_t index, sqlite_row_data &val) {
const char *result;
const char *type;
sqlite3_stmt *stmt = nullptr;
const char sql[] = "SELECT * FROM " sqlite3_table " WHERE ID=";
std::string _sql = std::string(sql) + std::to_string(index);
int32_t ret = sqlite3_prepare_v2(this->database, _sql.c_str(), _sql.size(),
&stmt, nullptr);
if (ret == SQLITE_OK) {
if (sqlite3_step(stmt) ==
SQLITE_ROW) // sqlite3 本身有一个虚拟缓存的区域 ,step 为执行语句
{
struct_sqlite_row(val, stmt);
}
sqlite3_finalize(stmt);
} else {
printf("Sqlite3Error:%s\n", sqlite3_errstr(ret));
}
return 0;
}
int32_t sqlite3_instance::add(uint32_t index, sqlite_row_data val) {
return 0; }
int32_t sqlite3_instance::modify(uint32_t index, sqlite_row_data val) {
return 0;
}
/** * @brief Destroy the sqlite3 sql::sqlite3 oh yeach * */
sqlite3_instance::~sqlite3_instance() {
int32_t ret = sqlite3_close(this->database);
if (ret < 0) {
printf("%s\n", sqlite3_errstr(ret));
}
}
void switch_val(sqlite_row_data &data) {
if (data.type == "int1") {
data.val.i8 = (int8_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "int2") {
data.val.i16 = (int16_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "int4") {
data.val.i32 = (int32_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "uint1") {
data.val.u8 = (uint8_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "uint2") {
data.val.u16 = (uint16_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "uint4") {
data.val.u32 = (uint32_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "str") {
memcpy(data.val.str, data._val.c_str(),
data._val.size() > 255 ? 255 : data._val.size());
} else if (data.type == "bit") {
data.val.bit = ((uint8_t)stoi(data._val, nullptr, 16)) & 0x1;
}
}
/** * @brief parser row data for virtual layer data * * @param index * @param data */
void str2sqlite_row_data(const char *ptr, int32_t index,
sqlite_row_data &data) {
switch (index) {
case ID:
data.id = (uint32_t)strtol(ptr, nullptr, 16);
break;
case NAME:
data.name = std::string(ptr);
break;
case MODE:
data.mode = std::string(ptr);
break;
case TYPE:
data.type = std::string(ptr);
break;
case COEF:
data.coeff = (uint32_t)strtol(ptr, nullptr, 10);
break;
case LEN:
data.len = (uint32_t)strtol(ptr, nullptr, 10);
break;
case VALUE:
data._val = std::string(ptr);
break;
case MIN:
data.min = (uint32_t)strtol(ptr, nullptr, 10);
break;
case MAX:
data.max = (uint32_t)strtol(ptr, nullptr, 10);
break;
case ATTR:
data.attr = (uint32_t)strtol(ptr, nullptr, 16); // NOTICE: ATTR use hex code
break;
}
}
/** * @brief parser item for virtual data * * @param pool * @param split * @param virtual_data */
void str2sqlite_row(char *pool, const char *split, sqlite_row_data &tmp) {
char *ptr;
int32_t i = 0;
while ((ptr = strsep(&pool, split)) != NULL) {
if (i == SQL_COL) {
break;
}
str2sqlite_row_data(ptr, i, tmp);
i++;
}
switch_val(tmp);
}
class virtual_cache {
private:
sqlite3_instance *sqlite;
std::map<uint32_t, sqlite_row_data> *cache;
const char *sqlite_config;
public:
explicit virtual_cache(sqlite3_instance *_sqlite,
std::map<uint32_t, sqlite_row_data> *_cache,
const char *sqlite_config);
int32_t set_data(uint32_t index, sqlite_row_data value);
int32_t get_data(uint32_t index, sqlite_row_data &value);
int32_t del_data(uint32_t index);
int32_t add_data(uint32_t index, sqlite_row_data value);
int32_t bind_event(uint32_t index, int32_t (*event)(void *), void *args);
~virtual_cache() {
sqlite = nullptr;
cache->erase(cache->begin(), cache->end());
#if defined(__GNUC__) && defined(__linux__)
malloc_trim(0); // FIXME: linux system used
#endif
}
};
virtual_cache::virtual_cache(sqlite3_instance *_sqlite,
std::map<uint32_t, sqlite_row_data> *_cache,
const char *_sqlite_config) {
if (_sqlite == nullptr || _cache == nullptr || _sqlite_config == nullptr) {
throw "init virtual_cache error";
}
this->sqlite = _sqlite;
this->cache = _cache;
this->sqlite_config = _sqlite_config;
}
int32_t virtual_cache::get_data(uint32_t index, sqlite_row_data &data) {
if (!cache->count(index)) {
return -1;
}
data = (*cache)[index];
return 0;
}
int32_t virtual_cache::set_data(uint32_t index, sqlite_row_data value) {
if (!cache->count(index)) {
return -1;
}
(*cache)[index] = value;
return 0;
}
int32_t virtual_cache::del_data(uint32_t index) {
if (!cache->count(index)) {
return -1;
}
cache->erase(index);
return 0;
}
int32_t virtual_cache::add_data(uint32_t index, sqlite_row_data value) {
cache->insert(std::make_pair(index, value));
return 0;
}
/** * @brief bind event normal exec sql operator * * @param index * @param event * @param args * @return int32_t */
int32_t virtual_cache::bind_event(uint32_t index, int32_t (*event)(void *),
void *args) {
if (!cache->count(index)) {
return -1;
}
(*cache)[index].event_binds.exec = event;
(*cache)[index].event_binds.args = args;
}
virtual_cache *cache_layer;
// index detect important
int main(void) {
try {
sqlite3_instance *sqlite =
new sqlite3_instance("/home/work/projects/Oran/union/drurau.db");
const char sqlite_config[] =
"/home/work/projects/Oran/union/device_data.csv";
std::map<uint32_t, sqlite_row_data> *data =
new std::map<uint32_t, sqlite_row_data>;
virtual_cache *cache = new virtual_cache(sqlite, data, sqlite_config);
cache_layer = cache;
// FILE* fp = fopen("/home/work/projects/Oran/union/device_data.csv","r");
// if(fp == nullptr){
// perror("");
// return -1;
// }
// std::vector<std::string> sql;
// char line[1024];
// bzero(line,1024);
// sqlite_row_data data;
// while(fgets(line,sizeof(line),fp) != NULL){
// line[strlen(line)-1] = 0; //del \n for sqlite
// str2sqlite_row(line,",",data);
// virtual_sqlite_layer.insert(std::make_pair(data.id,data));
// }
// for(auto it = virtual_sqlite_layer.begin() ; it !=
// virtual_sqlite_layer.end() ; it++){
// std::cout << std::hex << (*it).first << std::endl;
// }
} catch (const char *error) {
printf("Sqlite3Error:%s\n", error);
return 1;
}
return 0;
}
依靠 lds 脚本 中的 链接 方法
typedef int32_t (*driver_probe)(app_interface *app);
extern driver_probe driver_probe_start;
extern driver_probe driver_probe_end;
#define __init__ __attribute__((unused, section(".driver_probe")))
#define driver_register(func) driver_probe _fn_##func __init__ = func
typedef int32_t (*thread_probe)(const app_interface *const app);
extern thread_probe thread_probe_start;
extern thread_probe thread_probe_end;
#define __init_thread__ __attribute__((unused, section(".thread_probe")))
#define thread_register(func) thread_probe _fn_##func __init_thread__ = func
typedef int32_t (*event_probe)(const app_interface *const app);
extern event_probe event_probe_start;
extern event_probe event_probe_end;
#define __init_event__ __attribute__((unused, section(".event_probe")))
#define event_register(func) event_probe _fn_##func __init_event__ = func
inline void driver_init(app_interface *app) {
for (auto it = &driver_probe_start; it < &driver_probe_end; it++) {
(*it)(app);
}
}
inline void thread_init(const app_interface *const app) {
for (auto it = &thread_probe_start; it < &thread_probe_end; it++) {
(*it)(app);
}
}
inline void event_init(const app_interface *const app) {
for (auto it = &event_probe_start; it < &event_probe_end; it++) {
(*it)(app);
}
}
完整版
#include <cstddef>
#include <cstdint>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <fcntl.h>
#include <fstream>
#include <iostream>
#include <map>
#include <memory>
#include <sqlite3.h>
#include <string>
#include <sys/stat.h>
#include <type_traits>
#include <unistd.h>
#include <utility>
#include <vector>
// Web Or Desktop App also need sqlite3 info
// ID,name,mode,var_type,coefficient,len,V0,min integer,max integer,pro integer
// virtual data layer operator sqlite3
#define ID 0x0
#define NAME 0x1
#define MODE 0x2
#define TYPE 0x3
#define COEF 0x4
#define LEN 0x5
#define VALUE 0x6
#define MAX 0x7
#define MIN 0x8
#define ATTR 0x9
#define SQL_COL 10
#define sqlite3_table "rau"
#define SQLITE_FIRST_CREAT 0x4
#define SQLITE_TABLE_INIT_OK 0x2
#define SQLITE_TABLE_NOT_OK 0x8
static const char create_table_sql[] =
"CREATE TABLE " sqlite3_table "("
"ID UNSIGNED BIG INT PRIMARY KEY NOT NULL,"
"NAME TEXT NOT NULL,"
"MODE TEXT NOT NULL,"
"TYPE TEXT NOT NULL,"
"COEF INT NOT NULL,"
"LEN INTEGER NOT NULL,"
"VALUE TEXT NOT NULL,"
"MAX INTEGER NOT NULL,"
"MIN INTEGER NOT NULL,"
"ATTR INTEGER NOT NULL);";
static const char check_table_sql[] = "SELECT count(*) FROM sqlite_master"
" WHERE type='table'"
" AND name = \"" sqlite3_table "\"";
std::string str_format(const char *fmt, ...) {
int old_size = strlen(fmt);
std::unique_ptr<char[]> buf(new char[old_size]);
va_list ap;
va_start(ap, fmt);
int new_size = vsnprintf(buf.get(), old_size, fmt, ap);
va_end(ap);
if (new_size < 0) {
return "";
}
buf.reset(new char[new_size + 1]);
va_start(ap, fmt);
new_size = vsnprintf(buf.get(), new_size + 1, fmt, ap);
va_end(ap);
if (new_size < 0) {
return "";
}
return std::string(buf.get());
}
struct sqlite_row_data {
union {
uint32_t u32;
uint16_t u16;
uint8_t u8;
int32_t i32;
int16_t i16;
int8_t i8;
uint8_t bit : 1;
char str[256];
} val; // user programe function use
uint32_t id;
std::string name;
std::string mode;
std::string type;
std::string _val; // use sql to insert
uint32_t len;
uint32_t coeff;
uint32_t min;
uint32_t max;
uint32_t attr;
struct {
int32_t (*exec)(void *);
void *args;
} event_binds;
};
class sqlite3_instance {
private:
uint32_t table_ok = 0x0;
sqlite3 *database = nullptr;
const char *db_path = nullptr;
public:
explicit sqlite3_instance(const char *path);
int32_t connect();
int32_t init_virtual_sqlite(const char *config,
std::map<uint32_t, sqlite_row_data> &data);
int32_t add(uint32_t index, sqlite_row_data val);
int32_t modify(uint32_t index, sqlite_row_data val);
int32_t querry(uint32_t index, sqlite_row_data &val);
int32_t del(uint32_t index);
int32_t sql_exec(const char *sql);
~sqlite3_instance();
};
/** * @brief FIXME: add mutex * */
sqlite3_instance::sqlite3_instance(const char *path) {
if (path == nullptr) {
throw "ERROR:sqlite3 need database file path";
}
this->db_path = path;
}
int32_t sqlite3_instance::connect() {
sqlite3_stmt *stmt;
int32_t ret = sqlite3_open(this->db_path, &(this->database));
if (ret != SQLITE_OK) {
printf("WARN:%s\n", sqlite3_errstr(ret));
} else {
ret =
sqlite3_prepare_v2(this->database, check_table_sql, -1, &stmt, nullptr);
if (ret != SQLITE_OK) {
printf("Sqlite3Warn:%s\n", sqlite3_errstr(ret));
} else {
if (sqlite3_step(stmt) == SQLITE_ROW) {
table_ok = (uint32_t)sqlite3_column_int(stmt, 0);
}
if (!table_ok) {
ret = sqlite3_exec(this->database, create_table_sql, nullptr, nullptr,
nullptr);
if (ret != SQLITE_OK) {
printf("Sqlite3Error:%s\n", sqlite3_errstr(ret));
}
table_ok = SQLITE_FIRST_CREAT; // FIXME: first create table;
} else {
table_ok = SQLITE_TABLE_INIT_OK;
}
}
}
return 0;
}
int32_t sqlite3_instance::sql_exec(const char *sql) {
int32_t ret = sqlite3_exec(this->database, sql, nullptr, nullptr, nullptr);
if (ret != SQLITE_OK) {
printf("Sqlite3Error:%s\n", sqlite3_errstr(ret));
return ret;
}
return 0;
}
int32_t sqlite3_instance::del(uint32_t index) {
const char sql[] = "DELETE FROM " sqlite3_table " WHERE ID=";
std::string _sql = std::string(sql) + std::to_string(index);
int ret =
sqlite3_exec(this->database, _sql.c_str(), nullptr, nullptr, nullptr);
if (ret != SQLITE_OK) {
printf("WARN:%s\n", sqlite3_errstr(ret));
return ret;
}
return 0;
}
// string 可以完成值的赋值和拷贝
/** * @brief struct filed with struct sqlite3_stmt * * @param val * @param stmt */
void struct_sqlite_row(sqlite_row_data &val, sqlite3_stmt *stmt) {
val.id = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, ID),
nullptr, 16);
val.name = std::string((const char *)sqlite3_column_text(stmt, NAME));
val.type = std::string((const char *)sqlite3_column_text(stmt, TYPE));
val.attr = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, ATTR),
nullptr, 16);
val.coeff = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, COEF),
nullptr, 16);
val.len = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, LEN),
nullptr, 16);
val.mode = std::string((const char *)sqlite3_column_text(stmt, MODE));
val._val = std::string((const char *)sqlite3_column_text(stmt, VALUE));
val.max = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, MAX),
nullptr, 16);
val.min = (uint32_t)strtol((const char *)sqlite3_column_text(stmt, MIN),
nullptr, 16);
}
/** * @brief only one result for sqlite3 querry * * @tparam T * @param index * @param value * @return int32_t */
int32_t sqlite3_instance::querry(uint32_t index, sqlite_row_data &val) {
const char *result;
const char *type;
sqlite3_stmt *stmt = nullptr;
const char sql[] = "SELECT * FROM " sqlite3_table " WHERE ID=";
std::string _sql = std::string(sql) + std::to_string(index);
int32_t ret = sqlite3_prepare_v2(this->database, _sql.c_str(), _sql.size(),
&stmt, nullptr);
if (ret == SQLITE_OK) {
if (sqlite3_step(stmt) ==
SQLITE_ROW) // sqlite3 本身有一个虚拟缓存的区域 ,step 为执行语句
{
struct_sqlite_row(val, stmt);
}
sqlite3_finalize(stmt);
} else {
printf("Sqlite3Error:%s\n", sqlite3_errstr(ret));
}
return 0;
}
// TODO: Future may be add insert function
int32_t sqlite3_instance::add(uint32_t index, sqlite_row_data val) {
const char sql[] = "INSERT INTO " sqlite3_table " VALUES";
std::string exec_sql = str_format(
"%s(\'%ld\',\'%s\',\'%s\',\'%s\',\'%d\',\'%d\',\'%s\',\'%d\',\'%d\',\'%"
"d\')",
sql, val.id, val.name.c_str(), val.mode.c_str(), val.type.c_str(),
val.coeff, val.len, val._val.c_str(), val.min, val.max, val.attr);
int32_t ret =
sqlite3_exec(this->database, exec_sql.c_str(), nullptr, nullptr, nullptr);
if (ret != SQLITE_OK) {
printf("Sqlite3Error: %s\n", sqlite3_errstr(ret));
return ret;
}
return 0;
}
/** * @brief alert sqlite3 value interface * * @param index * @param val * @return int32_t */
int32_t sqlite3_instance::modify(
uint32_t index,
sqlite_row_data val) {
// TODO: current modify sqlite3 database value;
const char sql[] = "UPDATE " sqlite3_table " SET VALUE = ";
std::string _sql = std::string(sql) + val._val +
std::string(" WHERE ID == ") + std::to_string(val.id);
// std::cout << _sql << std::endl;
int32_t ret =
sqlite3_exec(this->database, _sql.c_str(), nullptr, nullptr, nullptr);
if (ret != SQLITE_OK) {
printf("Sqlite3Error:%s\n", sqlite3_errstr(ret));
return ret;
}
return 0;
}
/** * @brief Destroy the sqlite3 sql::sqlite3 oh yeach * */
sqlite3_instance::~sqlite3_instance() {
int32_t ret = sqlite3_close(this->database);
if (ret < 0) {
printf("%s\n", sqlite3_errstr(ret));
}
}
void switch_val(sqlite_row_data &data) {
if (data.type == "int1") {
data.val.i8 = (int8_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "int2") {
data.val.i16 = (int16_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "int4") {
data.val.i32 = (int32_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "uint1") {
data.val.u8 = (uint8_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "uint2") {
data.val.u16 = (uint16_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "uint4") {
data.val.u32 = (uint32_t)std::stoi(data._val, nullptr, 10);
} else if (data.type == "str") {
memcpy(data.val.str, data._val.c_str(),
data._val.size() > 255 ? 255 : data._val.size());
} else if (data.type == "bit") {
data.val.bit = ((uint8_t)stoi(data._val, nullptr, 16)) & 0x1;
}
}
/** * @brief parser row data for virtual layer data * * @param index * @param data */
void str2sqlite_row_data(const char *ptr, int32_t index,
sqlite_row_data &data) {
switch (index) {
case ID:
data.id = (uint32_t)strtol(ptr, nullptr, 16);
break;
case NAME:
data.name = std::string(ptr);
break;
case MODE:
data.mode = std::string(ptr);
break;
case TYPE:
data.type = std::string(ptr);
break;
case COEF:
data.coeff = (uint32_t)strtol(ptr, nullptr, 10);
break;
case LEN:
data.len = (uint32_t)strtol(ptr, nullptr, 10);
break;
case VALUE:
data._val = std::string(ptr);
break;
case MIN:
data.min = (uint32_t)strtol(ptr, nullptr, 10);
break;
case MAX:
data.max = (uint32_t)strtol(ptr, nullptr, 10);
break;
case ATTR:
data.attr = (uint32_t)strtol(ptr, nullptr, 16); // NOTICE: ATTR use hex code
break;
}
}
/** * @brief parser item for virtual data * * @param pool * @param split * @param virtual_data */
void str2sqlite_row(char *pool, const char *split, sqlite_row_data &tmp) {
char *ptr;
int32_t i = 0;
while ((ptr = strsep(&pool, split)) != NULL) {
if (i == SQL_COL) {
break;
}
str2sqlite_row_data(ptr, i, tmp);
i++;
}
switch_val(tmp);
}
/** * @brief FIXME: please remmber to free your buffer * * @param path * @param buffer * @return int32_t */
int32_t read_file2sqlite(const char *path,
std::map<uint32_t, sqlite_row_data> &data,
std::string &sql_str) {
if (path == nullptr) {
return -1;
}
char *tmp = new char[256];
struct stat stat;
sqlite_row_data sqlite_row;
if (!access(path, F_OK)) {
FILE *fp = fopen(path, "r");
if (fp == nullptr) {
printf("%s : %s\n", __func__, strerror(errno));
return -1;
}
/** * @brief FIXME: some data maybe right for sqlite data * bind default nullptr for every events * first head for sqlite3 sql using */
if (fgets(tmp, 256, fp) != nullptr) {
tmp[strlen(tmp) - 1] = 0;
str2sqlite_row(tmp, ",", sqlite_row);
sqlite_row.event_binds.exec = nullptr;
sqlite_row.event_binds.args = nullptr;
data.insert(std::make_pair(sqlite_row.id, sqlite_row));
sql_str =
str_format("SELECT %ld AS ID,\'%s\' AS NAME,\'%s\' AS MODE,\'%s\' "
"AS TYPE,%d AS COEF,%d AS "
"LEN,\'%s\' AS VALUE,%d AS MIN,%d AS MAX,%d AS ATTR",
sqlite_row.id, sqlite_row.name.c_str(),
sqlite_row.mode.c_str(), sqlite_row.type.c_str(),
sqlite_row.coeff, sqlite_row.len, sqlite_row._val.c_str(),
sqlite_row.min, sqlite_row.max, sqlite_row.attr);
}
bzero(tmp, 256);
while (fgets(tmp, 256, fp) != nullptr) {
tmp[strlen(tmp) - 1] = 0;
str2sqlite_row(tmp, ",", sqlite_row);
sqlite_row.event_binds.exec = nullptr;
sqlite_row.event_binds.args = nullptr;
data.insert(std::make_pair(sqlite_row.id, sqlite_row));
sql_str += str_format(
R"( UNION SELECT '%ld','%s','%s','%s','%d','%d','%s','%d','%d','%d')",
sqlite_row.id, sqlite_row.name.c_str(), sqlite_row.mode.c_str(),
sqlite_row.type.c_str(), sqlite_row.coeff, sqlite_row.len,
sqlite_row._val.c_str(), sqlite_row.min, sqlite_row.max,
sqlite_row.attr);
bzero(tmp, 256);
}
} else {
return -1; /*config file not ok will return error code*/
}
return 0;
}
/** * @brief config sqlite database and init virtual data layer * * @param config * @param data * @return int32_t */
int32_t sqlite3_instance::init_virtual_sqlite(
const char *config, std::map<uint32_t, sqlite_row_data> &data) {
const char insert_sql[] = "INSERT INTO " sqlite3_table " ";
std::string sql_str = "";
std::string insert_string_sql = "";
switch (this->table_ok) {
case SQLITE_FIRST_CREAT:
if (read_file2sqlite(config, data, sql_str) == 0) {
insert_string_sql = std::string(insert_sql) + sql_str;
// std::cout << insert_string_sql << std::endl;
this->sql_exec(insert_string_sql.c_str());
}
break;
case SQLITE_TABLE_INIT_OK:
const char sql_all_data[] = "SELECT * FROM " sqlite3_table ";";
sqlite_row_data sqlite_data;
sqlite3_stmt *stmt;
int32_t ret =
sqlite3_prepare_v2(this->database, sql_all_data, -1, &stmt, nullptr);
if (ret == SQLITE_OK) {
while (sqlite3_step(stmt) == SQLITE_ROW) {
sqlite_data.id = (uint32_t)sqlite3_column_int64(stmt, ID); /**/
sqlite_data.name =
std::string((const char *)sqlite3_column_text(stmt, NAME));
sqlite_data._val =
std::string((const char *)sqlite3_column_text(stmt, VALUE));
sqlite_data.mode =
std::string((const char *)sqlite3_column_text(stmt, MODE));
sqlite_data.attr = sqlite3_column_int(stmt, ATTR);
sqlite_data.type =
std::string((const char *)sqlite3_column_text(stmt, TYPE));
sqlite_data.min = sqlite3_column_int(stmt, MIN);
sqlite_data.max = sqlite3_column_int(stmt, MAX);
sqlite_data.len = sqlite3_column_int(stmt, LEN);
sqlite_data.coeff = sqlite3_column_int(stmt, COEF);
sqlite_data.event_binds.exec = nullptr;
sqlite_data.event_binds.args = nullptr;
switch_val(sqlite_data);
data.insert(std::make_pair(sqlite_data.id, sqlite_data));
}
} else {
printf("Sqlite3Error:%s\n", sqlite3_errstr(ret));
return -1;
}
break;
}
return 0;
}
class virtual_layer_operator {
public:
virtual int32_t set_data(uint32_t index, sqlite_row_data value) = 0;
virtual int32_t get_data(uint32_t index, sqlite_row_data &value) = 0;
virtual int32_t del_data(uint32_t index) = 0;
virtual int32_t add_data(uint32_t index, sqlite_row_data value) = 0;
};
class virtual_layer_events {
public:
virtual int32_t events(uint32_t index) = 0;
virtual int32_t bind_event(uint32_t index, int32_t (*event)(void *),
void *) = 0;
};
class virtual_cache : public virtual_layer_operator, public virtual_layer_events {
private:
sqlite3_instance *sqlite;
std::map<uint32_t, sqlite_row_data> *cache;
const char *sqlite_config;
public:
explicit virtual_cache(sqlite3_instance *_sqlite,
std::map<uint32_t, sqlite_row_data> *_cache,
const char *sqlite_config);
int32_t set_data(uint32_t index, sqlite_row_data value);
int32_t get_data(uint32_t index, sqlite_row_data &value);
int32_t del_data(uint32_t index);
int32_t add_data(uint32_t index, sqlite_row_data value);
virtual int32_t bind_event(uint32_t index, int32_t (*event)(void *),
void *args);
virtual int32_t events(uint32_t index);
~virtual_cache() {
if (sqlite != nullptr) {
delete sqlite;
}
sqlite = nullptr;
cache->erase(cache->begin(), cache->end());
if (cache != nullptr) {
delete cache;
}
cache = nullptr;
sqlite_config = nullptr;
}
};
virtual_cache::virtual_cache(sqlite3_instance *_sqlite,
std::map<uint32_t, sqlite_row_data> *_cache,
const char *_sqlite_config) {
if (_sqlite == nullptr || _cache == nullptr || _sqlite_config == nullptr) {
throw "init virtual_cache error: inivald args ... ...";
}
this->sqlite = _sqlite;
this->cache = _cache;
this->sqlite_config = _sqlite_config;
if (this->sqlite->connect() == 0) {
if (this->sqlite->init_virtual_sqlite(this->sqlite_config,
*(this->cache)) != 0) {
throw "init virtual sqlite error ... ...";
}
} else {
throw "virtual cache connect sqlite database error";
}
}
int32_t virtual_cache::get_data(uint32_t index, sqlite_row_data &data) {
if (!cache->count(index)) {
return -1;
}
data = (*cache)[index];
return 0;
}
int32_t virtual_cache::events(uint32_t index) {
if (!cache->count(index)) {
return -1;
}
sqlite_row_data *tmp = &(*cache)[index];
if (tmp->event_binds.exec == nullptr) {
return -2;
}
return tmp->event_binds.exec(tmp->event_binds.args);
}
int32_t virtual_cache::set_data(uint32_t index, sqlite_row_data value) {
if (!cache->count(index)) {
return -1;
}
(*cache)[index] = value;
int32_t ret = this->sqlite->modify(index, value);
if (ret != SQLITE_OK) {
return ret;
}
return 0;
}
int32_t virtual_cache::del_data(uint32_t index) {
if (!cache->count(index)) {
return -1;
}
cache->erase(index);
int32_t ret = this->sqlite->del(index);
if (ret != SQLITE_OK) {
return ret;
}
return 0;
}
virtual_cache *get_virtual_cache(void *args) {
return (virtual_cache *)args; }
/** * @brief current debug not add data for version * * @param index * @param value * @return int32_t */
int32_t virtual_cache::add_data(uint32_t index, sqlite_row_data value) {
if (!cache->count(index)) {
cache->insert(std::make_pair(index, value));
int32_t ret = this->sqlite->add(index, value);
if (ret != SQLITE_OK) {
return -2;
}
} else {
return -1;
}
return 0;
}
int32_t print_debug(void *args) {
printf("%s\n", (char *)args);
return 0;
}
static const struct {
uint32_t index;
int32_t (*exec)(void *);
void *args;
} probe_events[] = {
{
0x2, print_debug, (void *)"hello world"}, {
}};
/** * @brief bind event normal exec sql operator * * @param index * @param event * @param args * @return int32_t */
int32_t virtual_cache::bind_event(uint32_t index, int32_t (*event)(void *),
void *args) {
if (!cache->count(index)) {
return -1;
}
(*cache)[index].event_binds.exec = event;
(*cache)[index].event_binds.args = args;
return 0;
}
/** * @brief global cacher for all thread and driver * */
virtual_cache *cache_layer;
// resvered some information
// index detect important
int main(void) {
try {
sqlite3_instance *sqlite =
new sqlite3_instance("/home/work/projects/Oran/union/drurau.db");
const char sqlite_config[] =
"/home/work/projects/Oran/union/device_data.csv";
std::map<uint32_t, sqlite_row_data> *data =
new std::map<uint32_t, sqlite_row_data>;
virtual_cache *cache = new virtual_cache(sqlite, data, sqlite_config);
cache_layer = cache; // cache layer will release mem
sqlite_row_data test;
cache->get_data(0x2, test);
printf("index: %#x name: %s value: %d\n", test.id, test.name.c_str(),
test.val.u8);
for (int32_t i = 0; i < sizeof(probe_events) / sizeof(probe_events[0]);
i++) {
cache->bind_event(probe_events[i].index, probe_events[i].exec,
probe_events[i].args);
}
cache->events(0x2);
test.id = 0xffff1992;
test.name = "debug";
test.val.bit = 0;
test.type = "bit";
test.attr = 0x101;
test.min = 0x0;
test.max = 0xff;
test._val = std::to_string(0);
test.coeff = 1;
test.len = 4;
test.mode = "RW";
cache->add_data(test.id, test);
test._val = std::to_string(0x1);
test.val.bit = 1;
cache->set_data(test.id, test);
cache->get_data(test.id, test);
std::cout << test.name << " " << test._val << std::endl;
cache->del_data(test.id);
} catch (const char *error) {
printf(R"(VirtualDataLayerError:%s )",
error);
return 1;
}
return 0;
}
边栏推荐
猜你喜欢
随机推荐
Video Object Detection
v-model的使用
awk 统计平均 最大 最小值
Openharmony first experience (1)
华为WLAN技术:AP上线及相关模板的配置实验
Orthodontic MIA micro-implant anchorage technology China 10th anniversary exchange meeting was held in Shenyang
电脑一键重装系统内存完整性无法打开怎么办
How to add custom syntax to MySQL?
存储资源盘活系统助力新基建
Polygon zkEVM 基本概念
华为企业组网实例:VRRP+MSTP典型组网配置
c sqlite...
ERC20转账压缩
七夕福利!中奖名单:书籍免费送!
运维就业现状怎么样?技能要求高吗?
如何给MySQL添加自定义语法 ?
拥抱Cmake小朋友 简单又实用,但是不灵活
性能测试流程
红外图像滤波
[Sql brush topic] Query information data--Day1