2018-06-12 20:34:05 +00:00
|
|
|
#include <unistd.h>
|
2020-01-10 19:20:59 +00:00
|
|
|
#include <dlfcn.h>
|
2018-06-12 20:34:05 +00:00
|
|
|
#include <sys/stat.h>
|
|
|
|
|
2020-03-09 08:50:30 +00:00
|
|
|
#include <magisk.hpp>
|
|
|
|
#include <db.hpp>
|
2021-01-11 10:19:10 +00:00
|
|
|
#include <socket.hpp>
|
2022-05-12 09:03:42 +00:00
|
|
|
#include <base.hpp>
|
2018-06-12 20:34:05 +00:00
|
|
|
|
2022-03-28 09:05:09 +00:00
|
|
|
#define DB_VERSION 12
|
2018-10-28 18:49:04 +00:00
|
|
|
|
2019-03-06 13:16:12 +00:00
|
|
|
using namespace std;
|
2018-11-04 23:24:08 +00:00
|
|
|
|
2022-01-18 03:54:33 +00:00
|
|
|
struct sqlite3;
|
2020-01-10 19:20:59 +00:00
|
|
|
|
2019-03-06 13:16:12 +00:00
|
|
|
static sqlite3 *mDB = nullptr;
|
2018-11-04 23:24:08 +00:00
|
|
|
|
2022-01-18 03:54:33 +00:00
|
|
|
#define DBLOGV(...)
|
|
|
|
//#define DBLOGV(...) LOGD("magiskdb: " __VA_ARGS__)
|
|
|
|
|
2020-01-10 19:20:59 +00:00
|
|
|
// SQLite APIs
|
|
|
|
|
|
|
|
#define SQLITE_OPEN_READWRITE 0x00000002 /* Ok for sqlite3_open_v2() */
|
|
|
|
#define SQLITE_OPEN_CREATE 0x00000004 /* Ok for sqlite3_open_v2() */
|
|
|
|
#define SQLITE_OPEN_FULLMUTEX 0x00010000 /* Ok for sqlite3_open_v2() */
|
|
|
|
|
|
|
|
static int (*sqlite3_open_v2)(
|
2020-12-31 06:11:24 +00:00
|
|
|
const char *filename,
|
|
|
|
sqlite3 **ppDb,
|
|
|
|
int flags,
|
|
|
|
const char *zVfs);
|
2020-01-10 19:20:59 +00:00
|
|
|
static const char *(*sqlite3_errmsg)(sqlite3 *db);
|
|
|
|
static int (*sqlite3_close)(sqlite3 *db);
|
|
|
|
static void (*sqlite3_free)(void *v);
|
|
|
|
static int (*sqlite3_exec)(
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3 *db,
|
|
|
|
const char *sql,
|
|
|
|
int (*callback)(void*, int, char**, char**),
|
|
|
|
void *v,
|
|
|
|
char **errmsg);
|
2020-01-10 19:20:59 +00:00
|
|
|
|
|
|
|
// Internal Android linker APIs
|
|
|
|
|
|
|
|
static void (*android_get_LD_LIBRARY_PATH)(char *buffer, size_t buffer_size);
|
|
|
|
static void (*android_update_LD_LIBRARY_PATH)(const char *ld_library_path);
|
|
|
|
|
|
|
|
#define DLERR(ptr) if (!(ptr)) { \
|
2020-12-31 06:11:24 +00:00
|
|
|
LOGE("db: %s\n", dlerror()); \
|
|
|
|
return false; \
|
2020-01-10 19:20:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#define DLOAD(handle, arg) {\
|
2020-12-31 06:11:24 +00:00
|
|
|
auto f = dlsym(handle, #arg); \
|
|
|
|
DLERR(f) \
|
|
|
|
*(void **) &(arg) = f; \
|
2020-01-10 19:20:59 +00:00
|
|
|
}
|
|
|
|
|
2020-01-21 17:20:14 +00:00
|
|
|
#ifdef __LP64__
|
2021-02-14 06:16:38 +00:00
|
|
|
constexpr char apex_path[] = "/apex/com.android.runtime/lib64:/apex/com.android.art/lib64:/apex/com.android.i18n/lib64:";
|
2020-01-10 19:20:59 +00:00
|
|
|
#else
|
2020-10-11 20:41:29 +00:00
|
|
|
constexpr char apex_path[] = "/apex/com.android.runtime/lib:/apex/com.android.art/lib:/apex/com.android.i18n/lib:";
|
2020-01-10 19:20:59 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static int dl_init = 0;
|
|
|
|
|
|
|
|
static bool dload_sqlite() {
|
2020-12-31 06:11:24 +00:00
|
|
|
if (dl_init)
|
|
|
|
return dl_init > 0;
|
|
|
|
dl_init = -1;
|
|
|
|
|
|
|
|
auto sqlite = dlopen("libsqlite.so", RTLD_LAZY);
|
|
|
|
if (!sqlite) {
|
|
|
|
// Should only happen on Android 10+
|
|
|
|
auto dl = dlopen("libdl_android.so", RTLD_LAZY);
|
|
|
|
DLERR(dl);
|
|
|
|
|
|
|
|
DLOAD(dl, android_get_LD_LIBRARY_PATH);
|
|
|
|
DLOAD(dl, android_update_LD_LIBRARY_PATH);
|
|
|
|
|
|
|
|
// Inject APEX into LD_LIBRARY_PATH
|
|
|
|
char ld_path[4096];
|
|
|
|
memcpy(ld_path, apex_path, sizeof(apex_path));
|
|
|
|
constexpr int len = sizeof(apex_path) - 1;
|
|
|
|
android_get_LD_LIBRARY_PATH(ld_path + len, sizeof(ld_path) - len);
|
|
|
|
android_update_LD_LIBRARY_PATH(ld_path);
|
|
|
|
sqlite = dlopen("libsqlite.so", RTLD_LAZY);
|
|
|
|
|
|
|
|
// Revert LD_LIBRARY_PATH just in case
|
|
|
|
android_update_LD_LIBRARY_PATH(ld_path + len);
|
|
|
|
}
|
|
|
|
DLERR(sqlite);
|
|
|
|
|
|
|
|
DLOAD(sqlite, sqlite3_open_v2);
|
|
|
|
DLOAD(sqlite, sqlite3_errmsg);
|
|
|
|
DLOAD(sqlite, sqlite3_close);
|
|
|
|
DLOAD(sqlite, sqlite3_exec);
|
|
|
|
DLOAD(sqlite, sqlite3_free);
|
|
|
|
|
|
|
|
dl_init = 1;
|
|
|
|
return true;
|
2020-01-10 19:20:59 +00:00
|
|
|
}
|
|
|
|
|
2021-08-27 08:06:03 +00:00
|
|
|
int db_strings::get_idx(string_view key) const {
|
|
|
|
int idx = 0;
|
|
|
|
for (const char *k : DB_STRING_KEYS) {
|
|
|
|
if (key == k)
|
|
|
|
break;
|
|
|
|
++idx;
|
2020-12-31 06:11:24 +00:00
|
|
|
}
|
|
|
|
return idx;
|
2018-11-04 23:24:08 +00:00
|
|
|
}
|
|
|
|
|
2019-03-06 13:16:12 +00:00
|
|
|
db_settings::db_settings() {
|
2020-12-31 06:11:24 +00:00
|
|
|
// Default settings
|
|
|
|
data[ROOT_ACCESS] = ROOT_ACCESS_APPS_AND_ADB;
|
|
|
|
data[SU_MULTIUSER_MODE] = MULTIUSER_MODE_OWNER_ONLY;
|
|
|
|
data[SU_MNT_NS] = NAMESPACE_MODE_REQUESTER;
|
2021-09-12 19:40:34 +00:00
|
|
|
data[DENYLIST_CONFIG] = false;
|
2021-09-15 09:49:54 +00:00
|
|
|
data[ZYGISK_CONFIG] = false;
|
2018-11-04 23:24:08 +00:00
|
|
|
}
|
|
|
|
|
2021-08-27 08:06:03 +00:00
|
|
|
int db_settings::get_idx(string_view key) const {
|
|
|
|
int idx = 0;
|
|
|
|
for (const char *k : DB_SETTING_KEYS) {
|
|
|
|
if (key == k)
|
|
|
|
break;
|
|
|
|
++idx;
|
2020-12-31 06:11:24 +00:00
|
|
|
}
|
|
|
|
return idx;
|
2018-11-04 23:24:08 +00:00
|
|
|
}
|
|
|
|
|
2018-11-04 08:38:06 +00:00
|
|
|
static int ver_cb(void *ver, int, char **data, char **) {
|
2020-12-31 06:11:24 +00:00
|
|
|
*((int *) ver) = parse_int(data[0]);
|
|
|
|
return 0;
|
2018-06-12 20:34:05 +00:00
|
|
|
}
|
|
|
|
|
2018-11-16 08:20:30 +00:00
|
|
|
#define err_ret(e) if (e) return e;
|
2018-10-28 18:49:04 +00:00
|
|
|
|
2018-11-16 08:20:30 +00:00
|
|
|
static char *open_and_init_db(sqlite3 *&db) {
|
2020-12-31 06:11:24 +00:00
|
|
|
if (!dload_sqlite())
|
|
|
|
return strdup("Cannot load libsqlite.so");
|
|
|
|
|
|
|
|
int ret = sqlite3_open_v2(MAGISKDB, &db,
|
|
|
|
SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE | SQLITE_OPEN_FULLMUTEX, nullptr);
|
|
|
|
if (ret)
|
|
|
|
return strdup(sqlite3_errmsg(db));
|
2022-03-10 04:43:34 +00:00
|
|
|
int ver = 0;
|
2020-12-31 06:11:24 +00:00
|
|
|
bool upgrade = false;
|
2022-03-10 04:43:34 +00:00
|
|
|
char *err = nullptr;
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3_exec(db, "PRAGMA user_version", ver_cb, &ver, &err);
|
|
|
|
err_ret(err);
|
|
|
|
if (ver > DB_VERSION) {
|
|
|
|
// Don't support downgrading database
|
|
|
|
sqlite3_close(db);
|
2021-09-20 12:17:45 +00:00
|
|
|
return strdup("Downgrading database is not supported");
|
2020-12-31 06:11:24 +00:00
|
|
|
}
|
2022-03-28 07:59:16 +00:00
|
|
|
|
|
|
|
auto create_policy = [&] {
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3_exec(db,
|
|
|
|
"CREATE TABLE IF NOT EXISTS policies "
|
2022-03-28 09:05:09 +00:00
|
|
|
"(uid INT, policy INT, until INT, logging INT, "
|
|
|
|
"notification INT, PRIMARY KEY(uid))",
|
2020-12-31 06:11:24 +00:00
|
|
|
nullptr, nullptr, &err);
|
2022-03-28 07:59:16 +00:00
|
|
|
};
|
|
|
|
auto create_settings = [&] {
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3_exec(db,
|
|
|
|
"CREATE TABLE IF NOT EXISTS settings "
|
|
|
|
"(key TEXT, value INT, PRIMARY KEY(key))",
|
|
|
|
nullptr, nullptr, &err);
|
2022-03-28 07:59:16 +00:00
|
|
|
};
|
|
|
|
auto create_strings = [&] {
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3_exec(db,
|
|
|
|
"CREATE TABLE IF NOT EXISTS strings "
|
|
|
|
"(key TEXT, value TEXT, PRIMARY KEY(key))",
|
|
|
|
nullptr, nullptr, &err);
|
2022-03-28 07:59:16 +00:00
|
|
|
};
|
|
|
|
auto create_denylist = [&] {
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3_exec(db,
|
2022-03-28 07:59:16 +00:00
|
|
|
"CREATE TABLE IF NOT EXISTS denylist "
|
|
|
|
"(package_name TEXT, process TEXT, PRIMARY KEY(package_name, process))",
|
2020-12-31 06:11:24 +00:00
|
|
|
nullptr, nullptr, &err);
|
2022-03-28 07:59:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Database changelog:
|
|
|
|
//
|
|
|
|
// 0 - 6: DB stored in app private data. There are no longer any code in the project to
|
|
|
|
// migrate these data, so no need to take any of these versions into consideration.
|
2022-03-28 09:05:09 +00:00
|
|
|
// 7 : create table `hidelist` (process TEXT, PRIMARY KEY(process))
|
|
|
|
// 8 : add new column (package_name TEXT) to table `hidelist`
|
|
|
|
// 9 : rebuild table `hidelist` to change primary key (PRIMARY KEY(package_name, process))
|
|
|
|
// 10: remove table `logs`
|
|
|
|
// 11: remove table `hidelist` and create table `denylist` (same data structure)
|
|
|
|
// 12: rebuild table `policies` to drop column `package_name`
|
2022-03-28 07:59:16 +00:00
|
|
|
|
|
|
|
if (/* 0, 1, 2, 3, 4, 5, 6 */ ver <= 6) {
|
|
|
|
create_policy();
|
2020-12-31 06:11:24 +00:00
|
|
|
err_ret(err);
|
2022-03-28 07:59:16 +00:00
|
|
|
create_settings();
|
|
|
|
err_ret(err);
|
|
|
|
create_strings();
|
|
|
|
err_ret(err);
|
|
|
|
create_denylist();
|
|
|
|
err_ret(err);
|
|
|
|
|
|
|
|
// Directly jump to latest
|
|
|
|
ver = DB_VERSION;
|
2020-12-31 06:11:24 +00:00
|
|
|
upgrade = true;
|
|
|
|
}
|
2022-03-28 07:59:16 +00:00
|
|
|
if (ver == 7) {
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3_exec(db,
|
|
|
|
"BEGIN TRANSACTION;"
|
|
|
|
"ALTER TABLE hidelist RENAME TO hidelist_tmp;"
|
|
|
|
"CREATE TABLE IF NOT EXISTS hidelist "
|
|
|
|
"(package_name TEXT, process TEXT, PRIMARY KEY(package_name, process));"
|
|
|
|
"INSERT INTO hidelist SELECT process as package_name, process FROM hidelist_tmp;"
|
|
|
|
"DROP TABLE hidelist_tmp;"
|
|
|
|
"COMMIT;",
|
|
|
|
nullptr, nullptr, &err);
|
|
|
|
err_ret(err);
|
2022-03-28 07:59:16 +00:00
|
|
|
// Directly jump to version 9
|
2020-12-31 06:11:24 +00:00
|
|
|
ver = 9;
|
|
|
|
upgrade = true;
|
|
|
|
}
|
2022-03-28 07:59:16 +00:00
|
|
|
if (ver == 8) {
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3_exec(db,
|
|
|
|
"BEGIN TRANSACTION;"
|
|
|
|
"ALTER TABLE hidelist RENAME TO hidelist_tmp;"
|
|
|
|
"CREATE TABLE IF NOT EXISTS hidelist "
|
|
|
|
"(package_name TEXT, process TEXT, PRIMARY KEY(package_name, process));"
|
|
|
|
"INSERT INTO hidelist SELECT * FROM hidelist_tmp;"
|
|
|
|
"DROP TABLE hidelist_tmp;"
|
|
|
|
"COMMIT;",
|
|
|
|
nullptr, nullptr, &err);
|
|
|
|
err_ret(err);
|
|
|
|
ver = 9;
|
|
|
|
upgrade = true;
|
|
|
|
}
|
2022-03-28 07:59:16 +00:00
|
|
|
if (ver == 9) {
|
2020-12-31 06:11:24 +00:00
|
|
|
sqlite3_exec(db, "DROP TABLE IF EXISTS logs", nullptr, nullptr, &err);
|
|
|
|
err_ret(err);
|
|
|
|
ver = 10;
|
|
|
|
upgrade = true;
|
|
|
|
}
|
2022-03-28 07:59:16 +00:00
|
|
|
if (ver == 10) {
|
2021-09-12 19:40:34 +00:00
|
|
|
sqlite3_exec(db,
|
|
|
|
"DROP TABLE IF EXISTS hidelist;"
|
|
|
|
"DELETE FROM settings WHERE key='magiskhide';",
|
|
|
|
nullptr, nullptr, &err);
|
|
|
|
err_ret(err);
|
2022-03-28 07:59:16 +00:00
|
|
|
create_denylist();
|
|
|
|
err_ret(err);
|
2021-09-12 19:40:34 +00:00
|
|
|
ver = 11;
|
|
|
|
upgrade = true;
|
|
|
|
}
|
2022-03-28 09:05:09 +00:00
|
|
|
if (ver == 11) {
|
|
|
|
sqlite3_exec(db,
|
|
|
|
"BEGIN TRANSACTION;"
|
|
|
|
"ALTER TABLE policies RENAME TO policies_tmp;"
|
|
|
|
"CREATE TABLE IF NOT EXISTS policies "
|
|
|
|
"(uid INT, policy INT, until INT, logging INT, "
|
|
|
|
"notification INT, PRIMARY KEY(uid));"
|
|
|
|
"INSERT INTO policies "
|
|
|
|
"SELECT uid, policy, until, logging, notification FROM policies_tmp;"
|
|
|
|
"DROP TABLE policies_tmp;"
|
|
|
|
"COMMIT;",
|
|
|
|
nullptr, nullptr, &err);
|
|
|
|
err_ret(err);
|
|
|
|
ver = 12;
|
|
|
|
upgrade = true;
|
|
|
|
}
|
2020-12-31 06:11:24 +00:00
|
|
|
|
|
|
|
if (upgrade) {
|
|
|
|
// Set version
|
|
|
|
char query[32];
|
|
|
|
sprintf(query, "PRAGMA user_version=%d", ver);
|
|
|
|
sqlite3_exec(db, query, nullptr, nullptr, &err);
|
|
|
|
err_ret(err);
|
|
|
|
}
|
|
|
|
return nullptr;
|
2018-10-28 18:49:04 +00:00
|
|
|
}
|
|
|
|
|
2019-03-06 13:16:12 +00:00
|
|
|
char *db_exec(const char *sql) {
|
2022-03-10 04:43:34 +00:00
|
|
|
char *err = nullptr;
|
2020-12-31 06:11:24 +00:00
|
|
|
if (mDB == nullptr) {
|
|
|
|
err = open_and_init_db(mDB);
|
|
|
|
db_err_cmd(err,
|
|
|
|
// Open fails, remove and reconstruct
|
|
|
|
unlink(MAGISKDB);
|
|
|
|
err = open_and_init_db(mDB);
|
|
|
|
err_ret(err);
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (mDB) {
|
|
|
|
sqlite3_exec(mDB, sql, nullptr, nullptr, &err);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
return nullptr;
|
2018-10-27 21:54:48 +00:00
|
|
|
}
|
|
|
|
|
2021-08-27 08:06:03 +00:00
|
|
|
static int sqlite_db_row_callback(void *cb, int col_num, char **data, char **col_name) {
|
|
|
|
auto &func = *static_cast<const db_row_cb*>(cb);
|
|
|
|
db_row row;
|
|
|
|
for (int i = 0; i < col_num; ++i)
|
|
|
|
row[col_name[i]] = data[i];
|
|
|
|
return func(row) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
2019-03-06 13:16:12 +00:00
|
|
|
char *db_exec(const char *sql, const db_row_cb &fn) {
|
2022-03-10 04:43:34 +00:00
|
|
|
char *err = nullptr;
|
2020-12-31 06:11:24 +00:00
|
|
|
if (mDB == nullptr) {
|
|
|
|
err = open_and_init_db(mDB);
|
|
|
|
db_err_cmd(err,
|
|
|
|
// Open fails, remove and reconstruct
|
|
|
|
unlink(MAGISKDB);
|
|
|
|
err = open_and_init_db(mDB);
|
|
|
|
err_ret(err);
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (mDB) {
|
2021-08-27 08:06:03 +00:00
|
|
|
sqlite3_exec(mDB, sql, sqlite_db_row_callback, (void *) &fn, &err);
|
2020-12-31 06:11:24 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
return nullptr;
|
2018-06-12 20:34:05 +00:00
|
|
|
}
|
|
|
|
|
2019-03-06 13:16:12 +00:00
|
|
|
int get_db_settings(db_settings &cfg, int key) {
|
2022-03-10 04:43:34 +00:00
|
|
|
char *err = nullptr;
|
2020-12-31 06:11:24 +00:00
|
|
|
auto settings_cb = [&](db_row &row) -> bool {
|
|
|
|
cfg[row["key"]] = parse_int(row["value"]);
|
2022-01-18 03:54:33 +00:00
|
|
|
DBLOGV("query %s=[%s]\n", row["key"].data(), row["value"].data());
|
2020-12-31 06:11:24 +00:00
|
|
|
return true;
|
|
|
|
};
|
|
|
|
if (key >= 0) {
|
|
|
|
char query[128];
|
2021-08-27 08:06:03 +00:00
|
|
|
snprintf(query, sizeof(query), "SELECT * FROM settings WHERE key='%s'", DB_SETTING_KEYS[key]);
|
2020-12-31 06:11:24 +00:00
|
|
|
err = db_exec(query, settings_cb);
|
|
|
|
} else {
|
2021-08-27 08:06:03 +00:00
|
|
|
err = db_exec("SELECT * FROM settings", settings_cb);
|
2020-12-31 06:11:24 +00:00
|
|
|
}
|
|
|
|
db_err_cmd(err, return 1);
|
|
|
|
return 0;
|
2018-10-27 21:54:48 +00:00
|
|
|
}
|
|
|
|
|
2019-03-06 13:16:12 +00:00
|
|
|
int get_db_strings(db_strings &str, int key) {
|
2022-03-10 04:43:34 +00:00
|
|
|
char *err = nullptr;
|
2020-12-31 06:11:24 +00:00
|
|
|
auto string_cb = [&](db_row &row) -> bool {
|
|
|
|
str[row["key"]] = row["value"];
|
2022-01-18 03:54:33 +00:00
|
|
|
DBLOGV("query %s=[%s]\n", row["key"].data(), row["value"].data());
|
2020-12-31 06:11:24 +00:00
|
|
|
return true;
|
|
|
|
};
|
|
|
|
if (key >= 0) {
|
|
|
|
char query[128];
|
2021-08-27 08:06:03 +00:00
|
|
|
snprintf(query, sizeof(query), "SELECT * FROM strings WHERE key='%s'", DB_STRING_KEYS[key]);
|
2020-12-31 06:11:24 +00:00
|
|
|
err = db_exec(query, string_cb);
|
|
|
|
} else {
|
2021-08-27 08:06:03 +00:00
|
|
|
err = db_exec("SELECT * FROM strings", string_cb);
|
2020-12-31 06:11:24 +00:00
|
|
|
}
|
|
|
|
db_err_cmd(err, return 1);
|
|
|
|
return 0;
|
2018-06-12 20:34:05 +00:00
|
|
|
}
|
|
|
|
|
2018-11-16 08:20:30 +00:00
|
|
|
void exec_sql(int client) {
|
2020-12-31 06:11:24 +00:00
|
|
|
run_finally f([=]{ close(client); });
|
2021-01-12 08:07:48 +00:00
|
|
|
string sql = read_string(client);
|
|
|
|
char *err = db_exec(sql.data(), [client](db_row &row) -> bool {
|
2020-12-31 06:11:24 +00:00
|
|
|
string out;
|
|
|
|
bool first = true;
|
|
|
|
for (auto it : row) {
|
|
|
|
if (first) first = false;
|
|
|
|
else out += '|';
|
|
|
|
out += it.first;
|
|
|
|
out += '=';
|
|
|
|
out += it.second;
|
|
|
|
}
|
2021-01-12 08:07:48 +00:00
|
|
|
write_string(client, out);
|
2020-12-31 06:11:24 +00:00
|
|
|
return true;
|
|
|
|
});
|
|
|
|
write_int(client, 0);
|
|
|
|
db_err_cmd(err, return; );
|
2018-10-27 21:54:48 +00:00
|
|
|
}
|
2020-01-10 19:20:59 +00:00
|
|
|
|
|
|
|
bool db_err(char *e) {
|
2020-12-31 06:11:24 +00:00
|
|
|
if (e) {
|
|
|
|
LOGE("sqlite3_exec: %s\n", e);
|
|
|
|
sqlite3_free(e);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2020-01-10 19:20:59 +00:00
|
|
|
}
|