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
|
#define DATA_PUSH_BUSY_TIMEOUT 600000
sqlite3 *g_sim_db;
void sql_exec(int blob_cnt, const void **blobs, const int *blob_sizes, const char *sql_format, ...) {
assert(sql_format);
va_list args;
va_start(args, sql_format);
int sql_len = vsnprintf(NULL, 0, sql_format, args) + 1;
char *sql_str = malloc(sql_len);
assert(sql_str);
va_end(args);
va_start(args, sql_format);
vsprintf(sql_str, sql_format, args);
va_end(args);
int sql_res;
sqlite3_stmt *sql_stmt;
sql_res = sqlite3_prepare_v2(g_sim_db, sql_str, -1, &sql_stmt, NULL);
assert(sql_res == SQLITE_OK);
free(sql_str);
for (int i = 0; i < blob_cnt; ++i) {
assert(blobs[i]);
sql_res = sqlite3_bind_blob(sql_stmt, i + 1, blobs[i], blob_sizes[i], SQLITE_STATIC);
assert(sql_res == SQLITE_OK);
}
// Only handle SQLITE_BUSY error, in which case we retry the query.
// Setting 'journal_mode=wal;' should help prevent busy database errors.
while (true) {
sql_res = sqlite3_step(sql_stmt);
if (sql_res == SQLITE_DONE || sql_res == SQLITE_ROW) {
break;
}
log_warn("SQLite database returned error %d with message:", sql_res);
log_warn(sqlite3_errmsg(g_sim_db));
if (sql_res == SQLITE_BUSY) {
log_info("Will retry query...");
continue;
}
assert(false);
}
sqlite3_finalize(sql_stmt);
}
void sql_open(void) {
sqlite3_open(DATA_PUSH_PATH, &g_sim_db);
assert(g_sim_db);
// Install busy handler to retry transactions if DB is locked
sqlite3_busy_timeout(g_sim_db, DATA_PUSH_BUSY_TIMEOUT);
// Enable Write-Ahead Logging (WAL)
// This seems to help prevent DB locks when displaying live data.
// See: https://sqlite.org/wal.html
sql_exec(0, NULL, NULL, "pragma journal_mode=wal;");
}
void sql_close(void) {
assert(g_sim_db);
sqlite3_close(g_sim_db);
}
|