minetest-mapper-cpp/db-sqlite3.h
Rogier a270c6077b Add an option to prevent keeping sqlite3 database locked for too long
While prescanning a large database, minetestmapper may keep the database
locked for too long, causing minetest to complain, and even bail out
sometimes. Minetest will print warnings like:

	SQLite3 database has been locked for <duration>

The new option --sqlite3-limit-prescan-query-size limits the number
of records queried from the database in a single query, thus limiting
the duration of the lock.

If minetest kept the database locked for 1 second or more, *and* if
the database was modified in that time, minetestmapper will issue a warning,
and suggest using --sqlite3-limit-prescan-query-size.
2016-08-03 12:14:42 +02:00

62 lines
1.6 KiB
C++

#ifndef _DB_SQLITE3_H
#define _DB_SQLITE3_H
#include "db.h"
#include <sqlite3.h>
#if __cplusplus >= 201103L
#include <unordered_map>
#include <unordered_set>
#else
#include <map>
#include <set>
#endif
#include <string>
#include <sstream>
#include "types.h"
class DBSQLite3 : public DB {
#if __cplusplus >= 201103L
typedef std::unordered_map<int64_t, ustring> BlockCache;
typedef std::unordered_set<int64_t> BlockIdSet;
#else
typedef std::map<int64_t, ustring> BlockCache;
typedef std::set<int64_t> BlockIdSet;
#endif
public:
DBSQLite3(const std::string &mapdir);
virtual int getBlocksQueriedCount(void);
virtual int getBlocksReadCount(void);
virtual const BlockPosList &getBlockPosList();
virtual Block getBlockOnPos(const BlockPos &pos);
~DBSQLite3();
static void setLimitBlockListQuerySize(int count = -1);
static bool warnDatabaseLockDelay;
private:
static int m_blockListQuerySize;
static bool m_firstDatabaseInitialized;
int m_blocksQueriedCount;
int m_blocksReadCount;
sqlite3 *m_db;
sqlite3_stmt *m_dataVersionStatement;
sqlite3_stmt *m_blockPosListStatement;
sqlite3_stmt *m_blockOnPosStatement;
sqlite3_stmt *m_blockOnRowidStatement;
std::ostringstream m_getBlockSetStatementBlocks;
BlockCache m_blockCache;
BlockPosList m_blockPosList;
BlockIdSet m_blockIdSet; // temporary storage. Only used if m_blockListQuerySize > 0
uint64_t m_blockPosListQueryTime;
int64_t getDataVersion();
void prepareBlockOnPosStatement(void);
int getBlockPosListRows();
Block getBlockOnPosRaw(const BlockPos &pos);
void cacheBlocks(sqlite3_stmt *SQLstatement);
};
#endif // _DB_SQLITE3_H