Annotation of parser3/src/main/pa_db_connection.C, revision 1.12
1.1 parser 1: /** @file
1.11 paf 2: Parser: sql driver connection implementation.
1.1 parser 3:
4: Copyright (c) 2001 ArtLebedev Group (http://www.artlebedev.com)
5: Author: Alexander Petrosyan <paf@design.ru> (http://design.ru/paf)
6:
1.12 ! paf 7: $Id: pa_db_connection.C,v 1.11 2001/10/25 13:17:53 paf Exp $
1.1 parser 8: */
9:
10: #include "pa_config_includes.h"
11: #ifdef HAVE_LIBDB
12:
13: #include "pa_db_connection.h"
1.11 paf 14: #include "pa_db_table.h"
1.1 parser 15: #include "pa_exception.h"
1.11 paf 16: #include "pa_threads.h"
17: #include "pa_stack.h"
18: #include "pa_common.h"
1.1 parser 19:
1.11 paf 20: // consts
21:
22: const int EXPIRE_UNUSED_TABLE_SECONDS=60;
23: const int CHECK_EXPIRED_TABLES_SECONDS=EXPIRE_UNUSED_TABLE_SECONDS*2;
24:
25: const char *DB_WARNING1="Finding last valid log LSN";
1.10 parser 26:
1.11 paf 27: // callbacks
1.10 parser 28:
1.11 paf 29: static void db_paniccall(DB_ENV *dbenv, int error) {
30: throw Exception(0, 0,
31: 0,
32: "db_panic: %s (%d)",
33: strerror(error), error);
34: }
35:
36: static void db_errcall(const char *, char *buffer) {
37: // were it warning?
38: if(strncmp(buffer, DB_WARNING1, strlen(DB_WARNING1))==0)
39: return;
1.5 parser 40:
1.11 paf 41: throw Exception(0, 0,
42: 0,
43: "db_err: %s",
44: buffer);
45: }
1.5 parser 46:
1.11 paf 47: static void expire_table(Array::Item *value, void *info) {
48: DB_Table& table=*static_cast<DB_Table *>(value);
49: time_t older_dies=reinterpret_cast<time_t>(info);
1.5 parser 50:
1.11 paf 51: if(table.connected() && table.expired(older_dies))
52: table.disconnect();
53: }
54: static void expire_tables(const Hash::Key& key, Hash::Val *value, void *info) {
55: Stack& stack=*static_cast<Stack *>(value);
56: for(int i=0; i<=stack.top_index(); i++)
57: expire_table(stack.get(i), info);
58: }
1.5 parser 59:
1.3 parser 60: // DB_Connection
61:
1.11 paf 62: DB_Connection::DB_Connection(Pool& pool, const String& adb_home) : Pooled(pool),
63: time_used(0), fservices_pool(0),
64: fdb_home(adb_home), fconnected(false), errors(0),
65: table_cache(pool),
66: prev_expiration_pass_time(0) {
67: }
68:
69: void DB_Connection::connect() {
70: //_asm int 3;
71: memset(&dbenv, 0, sizeof(dbenv));
72: dbenv.db_paniccall=db_paniccall;
73: dbenv.db_errcall=db_errcall;
74:
75: char DB_DATA_DIR__VALUE[MAX_STRING];
76: char DB_LOG_DIR__VALUE[MAX_STRING];
77: char DB_TMP_DIR__VALUE[MAX_STRING];
78:
79: const char *db_home_cstr=fdb_home.cstr(String::UL_FILE_SPEC);
80:
81: snprintf(DB_DATA_DIR__VALUE, MAX_STRING, "DB_DATA_DIR %s", db_home_cstr);
82: snprintf(DB_LOG_DIR__VALUE, MAX_STRING, "DB_LOG_DIR %s", db_home_cstr);
83: snprintf(DB_TMP_DIR__VALUE, MAX_STRING, "DB_TMP_DIR %s", db_home_cstr);
84:
85: char *db_config[] = {
86: DB_DATA_DIR__VALUE,
87: DB_LOG_DIR__VALUE,
88: DB_TMP_DIR__VALUE,
89: 0
90: };
91:
92: u_int32_t flags=
93: DB_CREATE
94: | DB_INIT_MPOOL | DB_INIT_LOCK | DB_INIT_LOG | DB_INIT_TXN;
95:
1.12 ! paf 96: /* try {
1.11 paf 97: // 1: trying to open with SOFT RECOVER option set
98: check("db_appinit soft recover", &fdb_home, db_appinit(
99: db_home_cstr,
100: db_config,
101: &dbenv,
102: flags | DB_RECOVER
103: ));
104: } catch(...) {
1.12 ! paf 105: try {*/
1.11 paf 106: // 2: trying to open WITHOUT SOFT RECOVER option set
107: check("db_appinit no recover", &fdb_home, db_appinit(
108: db_home_cstr,
109: db_config,
110: &dbenv,
111: flags
112: ));
1.12 ! paf 113: /* } catch(...) {
1.11 paf 114: // 3: trying to open with FATAL RECOVER option set
115: check("db_appinit fatal recover", &fdb_home, db_appinit(
116: db_home_cstr,
117: db_config,
118: &dbenv,
119: flags | DB_RECOVER_FATAL
120: ));
121: }
1.12 ! paf 122: }*/
1.11 paf 123:
124: fconnected=true;
125: }
126:
127: void DB_Connection::disconnect() {
128: // close tables
129: table_cache.for_each(expire_tables,
130: reinterpret_cast<void *>(0/* =in the past = expire[close] all*/));
131:
132: // destroy connection data
133: check("db_appexit", &fdb_home, db_appexit(&dbenv)); fconnected=false;
134: }
135:
136:
1.1 parser 137: void DB_Connection::check(const char *operation, const String *source, int error) {
138: switch(error) {
139: case 0:
140: // no error
141: break;
142:
143: default:
1.11 paf 144: errors++;
1.1 parser 145: throw Exception(0, 0,
1.3 parser 146: source,
1.11 paf 147: "db %s error: %s (%d)",
148: operation, strerror(error), error);
1.1 parser 149: }
150: }
151:
1.11 paf 152: DB_Table& DB_Connection::get_table(const String& request_file_name,
153: const String& request_origin) {
154: Pool& pool=request_origin.pool(); // request pool
155:
156: // first trying to get cached table
157: DB_Table *result=get_table_from_cache(request_file_name);
158: if(result && !result->ping()) { // we have some cached table, is it pingable?
159: result->disconnect(); // kill unpingabe=dead table
160: result=0;
161: }
1.4 parser 162:
1.11 paf 163: char *request_file_name_cstr;
164: if(result)
165: request_file_name_cstr=0; // calm, compiler
166: else { // no cached table
167: // make global_file_name C-string on global pool
168: request_file_name_cstr=request_file_name.cstr(String::UL_AS_IS);
169: char *global_file_name_cstr=(char *)malloc(strlen(request_file_name_cstr)+1);
170: strcpy(global_file_name_cstr, request_file_name_cstr);
171: // make global_file_name string on global pool
172: String& global_file_name=*new(this->pool()) String(this->pool(), global_file_name_cstr);
173:
174: // allocate in global pool
175: // associate with services[request]
176: // NOTE: never freed up!
177: result=new(this->pool()) DB_Table(this->pool(), global_file_name, *this);
1.4 parser 178: }
1.1 parser 179:
1.11 paf 180: // associate with services[request] (deassociates at close)
181: result->set_services(&pool);
182: // if not connected yet, do that now, when result has services
183: if(!result->connected())
184: result->connect();
185: // return it
186: return *result;
187: }
188: void DB_Connection::clear_dbfile(const String& file_name) {
189: // open&clear
190: DB *db;
1.1 parser 191: DB_INFO dbinfo;
192: memset(&dbinfo, 0, sizeof(dbinfo));
1.11 paf 193: check("open(clear)", &file_name, db_open(
194: file_name.cstr(String::UL_FILE_SPEC),
1.1 parser 195: PA_DB_ACCESS_METHOD,
1.11 paf 196: DB_CREATE | DB_TRUNCATE/* used in single thread, no need for |DB_THREAD*/,
1.1 parser 197: 0666,
1.11 paf 198: &dbenv, &dbinfo, &db));
199:
200: check("close", &file_name, db->close(db, 0/*flags*/)); db=0;
1.1 parser 201: }
1.11 paf 202:
203: void DB_Connection::close_table(const String& file_name,
204: DB_Table& table) {
205: // deassociate from services[request]
206: table.set_services(0);
207: put_table_to_cache(file_name, table);
1.2 parser 208: }
1.1 parser 209:
210:
1.11 paf 211: // table cache
212: /// @todo get rid of memory spending Stack [zeros deep inside got accumulated]
213: DB_Table *DB_Connection::get_table_from_cache(const String& file_name) {
214: SYNCHRONIZED;
215:
216: maybe_expire_table_cache();
1.2 parser 217:
1.11 paf 218: if(Stack *tables=static_cast<Stack *>(table_cache.get(file_name)))
219: while(tables->top_index()>=0) { // there are cached tables to that 'file_name'
220: DB_Table *result=static_cast<DB_Table *>(tables->pop());
221: if(result->connected()) // not expired?
222: return result;
223: }
1.1 parser 224:
1.11 paf 225: return 0;
1.3 parser 226: }
227:
1.11 paf 228: void DB_Connection::put_table_to_cache(const String& file_name,
229: DB_Table& table) {
230: SYNCHRONIZED;
1.3 parser 231:
1.11 paf 232: Stack *tables=static_cast<Stack *>(table_cache.get(file_name));
233: if(!tables) { // there are no cached tables to that 'file_name' yet?
234: tables=NEW Stack(pool()); // NOTE: never freed up!
235: table_cache.put(file_name, tables);
236: }
237: tables->push(&table);
1.3 parser 238: }
239:
1.11 paf 240: void DB_Connection::maybe_expire_table_cache() {
241: time_t now=time(0);
242:
243: if(prev_expiration_pass_time<now-CHECK_EXPIRED_TABLES_SECONDS) {
244: table_cache.for_each(expire_tables,
245: reinterpret_cast<void *>(now-EXPIRE_UNUSED_TABLE_SECONDS));
1.3 parser 246:
1.11 paf 247: prev_expiration_pass_time=now;
1.9 parser 248: }
1.1 parser 249: }
250:
251: #endif
E-mail: