java_Db.c
Upload User: tsgydb
Upload Date: 2007-04-14
Package Size: 10674k
Code Size: 25k
Category:

MySQL

Development Platform:

Visual C++

  1. /*-
  2.  * See the file LICENSE for redistribution information.
  3.  *
  4.  * Copyright (c) 1997, 1998, 1999, 2000
  5.  * Sleepycat Software.  All rights reserved.
  6.  */
  7. #include "db_config.h"
  8. #ifndef lint
  9. static const char revid[] = "$Id: java_Db.c,v 11.34 2000/11/30 00:58:38 ubell Exp $";
  10. #endif /* not lint */
  11. #include <jni.h>
  12. #include <stdlib.h>
  13. #include <string.h>
  14. #include "db.h"
  15. #include "db_int.h"
  16. #include "db_page.h"
  17. #include "db_ext.h"
  18. #include "java_util.h"
  19. #include "com_sleepycat_db_Db.h"
  20. /* This struct is used in Db.verify and its callback */
  21. struct verify_callback_struct {
  22. JNIEnv *env;
  23. jobject streamobj;
  24. jbyteArray bytes;
  25. int nbytes;
  26. jmethodID writemid;
  27. };
  28. JAVADB_WO_ACCESS_METHOD(Db, jint, flags, DB, flags)
  29. JAVADB_WO_ACCESS_METHOD(Db, jint, h_1ffactor, DB, h_ffactor)
  30. JAVADB_WO_ACCESS_METHOD(Db, jint, h_1nelem, DB, h_nelem)
  31. JAVADB_WO_ACCESS_METHOD(Db, jint, lorder, DB, lorder)
  32. JAVADB_WO_ACCESS_METHOD(Db, jint, re_1delim, DB, re_delim)
  33. JAVADB_WO_ACCESS_METHOD(Db, jint, re_1len, DB, re_len)
  34. JAVADB_WO_ACCESS_METHOD(Db, jint, re_1pad, DB, re_pad)
  35. JAVADB_WO_ACCESS_METHOD(Db, jint, q_1extentsize, DB, q_extentsize)
  36. JAVADB_WO_ACCESS_METHOD(Db, jint, bt_1maxkey, DB, bt_maxkey)
  37. JAVADB_WO_ACCESS_METHOD(Db, jint, bt_1minkey, DB, bt_minkey)
  38. /* This only gets called once ever, at the beginning of execution
  39.  * and can be used to initialize unchanging methodIds, fieldIds, etc.
  40.  */
  41. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_one_1time_1init
  42.   (JNIEnv *jnienv,  /*Db.class*/ jclass jthisclass)
  43. {
  44. COMPQUIET(jnienv, NULL);
  45. COMPQUIET(jthisclass, NULL);
  46. }
  47. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db__1init
  48.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbEnv*/ jobject jdbenv, jint flags)
  49. {
  50. int err;
  51. DB *db;
  52. DB_JAVAINFO *dbinfo;
  53. DB_ENV *dbenv;
  54. dbenv = get_DB_ENV(jnienv, jdbenv);
  55. dbinfo = get_DB_JAVAINFO(jnienv, jthis);
  56. DB_ASSERT(dbinfo == NULL);
  57. err = db_create(&db, dbenv, flags);
  58. if (verify_return(jnienv, err, 0)) {
  59. set_private_dbobj(jnienv, name_DB, jthis, db);
  60. dbinfo = dbji_construct(jnienv, flags);
  61. set_private_info(jnienv, name_DB, jthis, dbinfo);
  62. db->cj_internal = dbinfo;
  63. }
  64. }
  65. JNIEXPORT jint JNICALL Java_com_sleepycat_db_Db__1close
  66.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jint flags)
  67. {
  68. int err;
  69. DB *db;
  70. DB_JAVAINFO *dbinfo;
  71. db = get_DB(jnienv, jthis);
  72. dbinfo = get_DB_JAVAINFO(jnienv, jthis);
  73. if (!verify_non_null(jnienv, db))
  74. return (0);
  75. JAVADB_API_BEGIN(db, jthis);
  76. /* Null out the private data to indicate the DB is invalid.
  77.  * We do this in advance to help guard against multithreading
  78.  * issues.
  79.  */
  80. set_private_dbobj(jnienv, name_DB, jthis, 0);
  81. err = db->close(db, flags);
  82. if (err != DB_INCOMPLETE)
  83. verify_return(jnienv, err, 0);
  84. dbji_dealloc(dbinfo, jnienv);
  85. /* don't call JAVADB_API_END - db cannot be used */
  86. return (err);
  87. }
  88. /* We are being notified that the parent DbEnv has closed.
  89.  * Zero out the pointer to the DB, since it is no longer
  90.  * valid, to prevent mistakes.  The user will get a null
  91.  * pointer exception if they try to use this Db again.
  92.  */
  93. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db__1notify_1internal
  94.   (JNIEnv *jnienv, /*Db*/ jobject jthis)
  95. {
  96. set_private_dbobj(jnienv, name_DB, jthis, 0);
  97. }
  98. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_append_1recno_1changed
  99.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbAppendRecno*/ jobject jcallback)
  100. {
  101. DB *db;
  102. DB_JAVAINFO *dbinfo;
  103. db = get_DB(jnienv, jthis);
  104. if (!verify_non_null(jnienv, db))
  105. return;
  106. JAVADB_API_BEGIN(db, jthis);
  107. dbinfo = (DB_JAVAINFO*)db->cj_internal;
  108. dbji_set_append_recno_object(dbinfo, jnienv, db, jcallback);
  109. JAVADB_API_END(db);
  110. }
  111. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_bt_1compare_1changed
  112.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbBtreeCompare*/ jobject jbtcompare)
  113. {
  114. DB *db;
  115. DB_JAVAINFO *dbinfo;
  116. db = get_DB(jnienv, jthis);
  117. if (!verify_non_null(jnienv, db))
  118. return;
  119. JAVADB_API_BEGIN(db, jthis);
  120. dbinfo = (DB_JAVAINFO*)db->cj_internal;
  121. dbji_set_bt_compare_object(dbinfo, jnienv, db, jbtcompare);
  122. JAVADB_API_END(db);
  123. }
  124. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_bt_1prefix_1changed
  125.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbBtreePrefix*/ jobject jbtprefix)
  126. {
  127. DB *db;
  128. DB_JAVAINFO *dbinfo;
  129. db = get_DB(jnienv, jthis);
  130. if (!verify_non_null(jnienv, db))
  131. return;
  132. JAVADB_API_BEGIN(db, jthis);
  133. dbinfo = (DB_JAVAINFO*)db->cj_internal;
  134. dbji_set_bt_prefix_object(dbinfo, jnienv, db, jbtprefix);
  135. JAVADB_API_END(db);
  136. }
  137. JNIEXPORT jobject JNICALL Java_com_sleepycat_db_Db_cursor
  138.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbTxn*/ jobject txnid, jint flags)
  139. {
  140. int err;
  141. DBC *dbc;
  142. DB *db = get_DB(jnienv, jthis);
  143. DB_TXN *dbtxnid = get_DB_TXN(jnienv, txnid);
  144. if (!verify_non_null(jnienv, db))
  145. return (NULL);
  146. err = db->cursor(db, dbtxnid, &dbc, flags);
  147. verify_return(jnienv, err, 0);
  148. return (get_Dbc(jnienv, dbc));
  149. }
  150. JNIEXPORT jint JNICALL Java_com_sleepycat_db_Db_del
  151.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbTxn*/ jobject txnid,
  152.    /*Dbt*/ jobject key, jint dbflags)
  153. {
  154. int err;
  155. DB_TXN *dbtxnid;
  156. DB *db;
  157. JDBT dbkey;
  158. err = 0;
  159. db = get_DB(jnienv, jthis);
  160. if (!verify_non_null(jnienv, db))
  161. return (0);
  162. JAVADB_API_BEGIN(db, jthis);
  163. dbtxnid = get_DB_TXN(jnienv, txnid);
  164. if (jdbt_lock(&dbkey, jnienv, key, inOp) != 0)
  165. goto out;
  166. err = db->del(db, dbtxnid, &dbkey.dbt->dbt, dbflags);
  167. if (err != DB_NOTFOUND) {
  168. verify_return(jnienv, err, 0);
  169. }
  170.  out:
  171. jdbt_unlock(&dbkey, jnienv);
  172. JAVADB_API_END(db);
  173. return (err);
  174. }
  175. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_dup_1compare_1changed
  176.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbDupCompare*/ jobject jdupcompare)
  177. {
  178. DB *db;
  179. DB_JAVAINFO *dbinfo;
  180. db = get_DB(jnienv, jthis);
  181. if (!verify_non_null(jnienv, db))
  182. return;
  183. JAVADB_API_BEGIN(db, jthis);
  184. dbinfo = (DB_JAVAINFO*)db->cj_internal;
  185. dbji_set_dup_compare_object(dbinfo, jnienv, db, jdupcompare);
  186. JAVADB_API_END(db);
  187. }
  188. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_err
  189.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jint ecode, jstring msg)
  190. {
  191. DB *db;
  192. JSTR msg_string;
  193. if (jstr_lock(&msg_string, jnienv, msg) != 0)
  194. goto out;
  195. db = get_DB(jnienv, jthis);
  196. if (!verify_non_null(jnienv, db))
  197. goto out;
  198. JAVADB_API_BEGIN(db, jthis);
  199. db->err(db, ecode, msg_string.string);
  200. JAVADB_API_END(db);
  201.  out:
  202. jstr_unlock(&msg_string, jnienv);
  203. }
  204. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_errx
  205.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jstring msg)
  206. {
  207. JSTR msg_string;
  208. DB *db = get_DB(jnienv, jthis);
  209. if (jstr_lock(&msg_string, jnienv, msg) != 0)
  210. goto out;
  211. if (!verify_non_null(jnienv, db))
  212. goto out;
  213. JAVADB_API_BEGIN(db, jthis);
  214. db->errx(db, msg_string.string);
  215. JAVADB_API_END(db);
  216.  out:
  217. jstr_unlock(&msg_string, jnienv);
  218. }
  219. JNIEXPORT jint JNICALL Java_com_sleepycat_db_Db_fd
  220.   (JNIEnv *jnienv, /*Db*/ jobject jthis)
  221. {
  222. int err;
  223. int return_value = 0;
  224. DB *db = get_DB(jnienv, jthis);
  225. if (!verify_non_null(jnienv, db))
  226. return (0);
  227. JAVADB_API_BEGIN(db, jthis);
  228. err = db->fd(db, &return_value);
  229. verify_return(jnienv, err, 0);
  230. JAVADB_API_END(db);
  231. return (return_value);
  232. }
  233. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_feedback_1changed
  234.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbFeedback*/ jobject jfeedback)
  235. {
  236. DB *db;
  237. DB_JAVAINFO *dbinfo;
  238. db = get_DB(jnienv, jthis);
  239. if (!verify_non_null(jnienv, db))
  240. return;
  241. JAVADB_API_BEGIN(db, jthis);
  242. dbinfo = (DB_JAVAINFO*)db->cj_internal;
  243. dbji_set_feedback_object(dbinfo, jnienv, db, jfeedback);
  244. JAVADB_API_END(db);
  245. }
  246. JNIEXPORT jint JNICALL Java_com_sleepycat_db_Db_get
  247.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbTxn*/ jobject txnid,
  248.    /*Dbt*/ jobject key, /*Dbt*/ jobject data, jint flags)
  249. {
  250. int err, op_flags, retry;
  251. DB *db;
  252. OpKind keyop, dataop;
  253. DB_TXN *dbtxnid;
  254. JDBT dbkey, dbdata;
  255. err = 0;
  256. db = get_DB(jnienv, jthis);
  257. if (!verify_non_null(jnienv, db))
  258. goto out3;
  259. JAVADB_API_BEGIN(db, jthis);
  260. /* Depending on flags, the key may be input/output. */
  261. keyop = inOp;
  262. dataop = outOp;
  263. op_flags = flags & DB_OPFLAGS_MASK;
  264. if (op_flags == DB_SET_RECNO) {
  265. keyop = inOutOp;
  266. }
  267. else if (op_flags == DB_GET_BOTH) {
  268. keyop = inOutOp;
  269. dataop = inOutOp;
  270. }
  271. dbtxnid = get_DB_TXN(jnienv, txnid);
  272. if (jdbt_lock(&dbkey, jnienv, key, keyop) != 0)
  273. goto out2;
  274. if (jdbt_lock(&dbdata, jnienv, data, dataop) != 0)
  275. goto out1;
  276. for (retry = 0; retry < 3; retry++) {
  277. err = db->get(db, dbtxnid, &dbkey.dbt->dbt, &dbdata.dbt->dbt, flags);
  278. /* If we failed due to lack of memory in our DBT arrays,
  279.  * retry.
  280.  */
  281. if (err != ENOMEM)
  282. break;
  283. if (!jdbt_realloc(&dbdata, jnienv))
  284. break;
  285. }
  286. if (err != DB_NOTFOUND) {
  287. verify_return(jnienv, err, 0);
  288. }
  289.  out1:
  290. jdbt_unlock(&dbdata, jnienv);
  291.  out2:
  292. jdbt_unlock(&dbkey, jnienv);
  293.  out3:
  294. JAVADB_API_END(db);
  295. return (err);
  296. }
  297. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_hash_1changed
  298.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbHash*/ jobject jhash)
  299. {
  300. DB *db;
  301. DB_JAVAINFO *dbinfo;
  302. db = get_DB(jnienv, jthis);
  303. if (!verify_non_null(jnienv, db))
  304. return;
  305. JAVADB_API_BEGIN(db, jthis);
  306. dbinfo = (DB_JAVAINFO*)db->cj_internal;
  307. dbji_set_h_hash_object(dbinfo, jnienv, db, jhash);
  308. JAVADB_API_END(db);
  309. }
  310. JNIEXPORT jobject JNICALL Java_com_sleepycat_db_Db_join
  311.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*Dbc[]*/ jobjectArray curslist,
  312.    jint flags)
  313. {
  314. int err;
  315. DB *db = get_DB(jnienv, jthis);
  316. int count = (*jnienv)->GetArrayLength(jnienv, curslist);
  317. DBC **newlist = (DBC **)malloc(sizeof(DBC *) * (count+1));
  318. DBC *dbc;
  319. int i;
  320. /* Convert the java array of Dbc's to a C array of DBC's. */
  321. for (i=0; i<count; i++) {
  322. jobject jobj = (*jnienv)->GetObjectArrayElement(jnienv, curslist, i);
  323. if (jobj == 0) {
  324. /*
  325.  * An embedded null in the array is treated
  326.  * as an endpoint.
  327.  */
  328. newlist[i] = 0;
  329. break;
  330. }
  331. else {
  332. newlist[i] = get_DBC(jnienv, jobj);
  333. }
  334. }
  335. newlist[count] = 0;
  336. if (!verify_non_null(jnienv, db))
  337. return (NULL);
  338. JAVADB_API_BEGIN(db, jthis);
  339. err = db->join(db, newlist, &dbc, flags);
  340. free(newlist);
  341. verify_return(jnienv, err, 0);
  342. JAVADB_API_END(db);
  343. return (get_Dbc(jnienv, dbc));
  344. }
  345. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_key_1range
  346.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbTxn*/ jobject jtxn,
  347.    /*Dbt*/ jobject jkey, jobject /*DbKeyRange*/ range, jint flags)
  348. {
  349. int err;
  350. DB *db = get_DB(jnienv, jthis);
  351. DB_TXN *txn = get_DB_TXN(jnienv, jtxn);
  352. JDBT dbkey;
  353. DB_KEY_RANGE result;
  354. jfieldID fid;
  355. jclass krclass;
  356. if (!verify_non_null(jnienv, db))
  357. return;
  358. JAVADB_API_BEGIN(db, jthis);
  359. if (!verify_non_null(jnienv, range))
  360. return;
  361. if (jdbt_lock(&dbkey, jnienv, jkey, inOp) != 0)
  362. goto out;
  363. err = db->key_range(db, txn, &dbkey.dbt->dbt, &result, flags);
  364. if (verify_return(jnienv, err, 0)) {
  365. /* fill in the values of the DbKeyRange structure */
  366. krclass = get_class(jnienv, "DbKeyRange");
  367. fid = (*jnienv)->GetFieldID(jnienv, krclass, "less", "D");
  368. (*jnienv)->SetDoubleField(jnienv, range, fid, result.less);
  369. fid = (*jnienv)->GetFieldID(jnienv, krclass, "equal", "D");
  370. (*jnienv)->SetDoubleField(jnienv, range, fid, result.equal);
  371. fid = (*jnienv)->GetFieldID(jnienv, krclass, "greater", "D");
  372. (*jnienv)->SetDoubleField(jnienv, range, fid, result.greater);
  373. }
  374.  out:
  375. jdbt_unlock(&dbkey, jnienv);
  376. JAVADB_API_END(db);
  377. }
  378. JNIEXPORT jint JNICALL Java_com_sleepycat_db_Db_put
  379.   (JNIEnv *jnienv, /*Db*/ jobject jthis, /*DbTxn*/ jobject txnid,
  380.    /*Dbt*/ jobject key, /*Dbt*/ jobject data, jint flags)
  381. {
  382. int err;
  383. DB *db;
  384. DB_TXN *dbtxnid;
  385. JDBT dbkey, dbdata;
  386. err = 0;
  387. db = get_DB(jnienv, jthis);
  388. dbtxnid = get_DB_TXN(jnienv, txnid);
  389. if (!verify_non_null(jnienv, db))
  390. return (0);    /* error will be thrown, retval doesn't matter */
  391. JAVADB_API_BEGIN(db, jthis);
  392. if (jdbt_lock(&dbkey, jnienv, key, inOp) != 0)
  393. goto out2;
  394. if (jdbt_lock(&dbdata, jnienv, data, inOp) != 0)
  395. goto out1;
  396. if (!verify_non_null(jnienv, db))
  397. goto out1;
  398. err = db->put(db, dbtxnid, &dbkey.dbt->dbt, &dbdata.dbt->dbt, flags);
  399. if (err != DB_KEYEXIST) {
  400. verify_return(jnienv, err, 0);
  401. }
  402.  out1:
  403. jdbt_unlock(&dbdata, jnienv);
  404.  out2:
  405. jdbt_unlock(&dbkey, jnienv);
  406. JAVADB_API_END(db);
  407. return (err);
  408. }
  409. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_rename
  410.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jstring file,
  411.    jstring database, jstring newname, jint flags)
  412. {
  413. int err;
  414. DB *db;
  415. DB_JAVAINFO *dbinfo;
  416. JSTR j_file;
  417. JSTR j_database;
  418. JSTR j_newname;
  419. db = get_DB(jnienv, jthis);
  420. dbinfo = get_DB_JAVAINFO(jnienv, jthis);
  421. if (!verify_non_null(jnienv, db))
  422. return;
  423. JAVADB_API_BEGIN(db, jthis);
  424. if (jstr_lock(&j_file, jnienv, file) != 0)
  425. goto out3;
  426. if (jstr_lock(&j_database, jnienv, database) != 0)
  427. goto out2;
  428. if (jstr_lock(&j_newname, jnienv, newname) != 0)
  429. goto out1;
  430. err = db->rename(db, j_file.string, j_database.string,
  431.  j_newname.string, flags);
  432. verify_return(jnienv, err, EXCEPTION_FILE_NOT_FOUND);
  433. dbji_dealloc(dbinfo, jnienv);
  434. set_private_dbobj(jnienv, name_DB, jthis, 0);
  435.  out1:
  436. jstr_unlock(&j_newname, jnienv);
  437.  out2:
  438. jstr_unlock(&j_database, jnienv);
  439.  out3:
  440. jstr_unlock(&j_file, jnienv);
  441. /* don't call JAVADB_API_END - db cannot be used */
  442. }
  443. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_remove
  444.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jstring file,
  445.    jstring database, jint flags)
  446. {
  447. int err;
  448. DB *db = get_DB(jnienv, jthis);
  449. DB_JAVAINFO *dbinfo = get_DB_JAVAINFO(jnienv, jthis);
  450. JSTR j_file;
  451. JSTR j_database;
  452. dbinfo = get_DB_JAVAINFO(jnienv, jthis);
  453. if (!verify_non_null(jnienv, db))
  454. return;
  455. JAVADB_API_BEGIN(db, jthis);
  456. if (jstr_lock(&j_file, jnienv, file) != 0)
  457. goto out2;
  458. if (jstr_lock(&j_database, jnienv, database) != 0)
  459. goto out1;
  460. err = db->remove(db, j_file.string, j_database.string, flags);
  461. set_private_dbobj(jnienv, name_DB, jthis, 0);
  462. verify_return(jnienv, err, EXCEPTION_FILE_NOT_FOUND);
  463. dbji_dealloc(dbinfo, jnienv);
  464.  out1:
  465. jstr_unlock(&j_database, jnienv);
  466.  out2:
  467. jstr_unlock(&j_file, jnienv);
  468. /* don't call JAVADB_API_END - db cannot be used */
  469. }
  470. JNIEXPORT void JNICALL
  471.   Java_com_sleepycat_db_Db_set_1pagesize
  472.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jlong value)
  473. {
  474. int err;
  475. DB *db;
  476. db = get_DB(jnienv, jthis);
  477. if (verify_non_null(jnienv, db)) {
  478. JAVADB_API_BEGIN(db, jthis);
  479. err = db->set_pagesize(db, (u_int32_t)value);
  480. verify_return(jnienv, err, 0);
  481. JAVADB_API_END(db);
  482. }
  483. }
  484. JNIEXPORT void JNICALL
  485.   Java_com_sleepycat_db_Db_set_1cachesize
  486.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jint gbytes, jint bytes,
  487.    jint ncaches)
  488. {
  489. int err;
  490. DB *db;
  491. db = get_DB(jnienv, jthis);
  492. if (verify_non_null(jnienv, db)) {
  493. JAVADB_API_BEGIN(db, jthis);
  494. err = db->set_cachesize(db, gbytes, bytes, ncaches);
  495. verify_return(jnienv, err, 0);
  496. JAVADB_API_END(db);
  497. }
  498. }
  499. JNIEXPORT void JNICALL
  500.   Java_com_sleepycat_db_Db_set_1re_1source
  501.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jstring re_source)
  502. {
  503. int err;
  504. DB *db;
  505. db = get_DB(jnienv, jthis);
  506. if (verify_non_null(jnienv, db)) {
  507. JAVADB_API_BEGIN(db, jthis);
  508. /* XXX does the string from get_c_string ever get freed? */
  509. if (re_source != NULL)
  510. err = db->set_re_source(db, get_c_string(jnienv, re_source));
  511. else
  512. err = db->set_re_source(db, 0);
  513. verify_return(jnienv, err, 0);
  514. JAVADB_API_END(db);
  515. }
  516. }
  517. JNIEXPORT jobject JNICALL Java_com_sleepycat_db_Db_stat
  518.   (JNIEnv *jnienv, jobject jthis, jint flags)
  519. {
  520. int err;
  521. DB *db = get_DB(jnienv, jthis);
  522. jobject retval = NULL;
  523. jclass dbclass;
  524. void *statp = 0;
  525. DB_BTREE_STAT *bstp;
  526. DB_HASH_STAT *hstp;
  527. DB_QUEUE_STAT *qstp;
  528. if (!verify_non_null(jnienv, db))
  529. return (NULL);
  530. JAVADB_API_BEGIN(db, jthis);
  531. err = db->stat(db, &statp, NULL, flags);
  532. if (verify_return(jnienv, err, 0)) {
  533. DBTYPE dbtype = db->get_type(db);
  534. switch (dbtype) {
  535. /* Btree and recno share the same stat structure */
  536. case DB_BTREE:
  537. case DB_RECNO:
  538. bstp = (DB_BTREE_STAT *)statp;
  539. retval = create_default_object(jnienv,
  540.        name_DB_BTREE_STAT);
  541. dbclass = get_class(jnienv, name_DB_BTREE_STAT);
  542. /* Set the individual fields */
  543. set_int_field(jnienv, dbclass, retval,
  544.       "bt_magic", bstp->bt_magic);
  545. set_int_field(jnienv, dbclass, retval,
  546.       "bt_version", bstp->bt_version);
  547. set_int_field(jnienv, dbclass, retval,
  548.       "bt_metaflags", bstp->bt_metaflags);
  549. set_int_field(jnienv, dbclass, retval,
  550.       "bt_nkeys", bstp->bt_nkeys);
  551. set_int_field(jnienv, dbclass, retval,
  552.       "bt_ndata", bstp->bt_ndata);
  553. set_int_field(jnienv, dbclass, retval,
  554.       "bt_pagesize", bstp->bt_pagesize);
  555. set_int_field(jnienv, dbclass, retval,
  556.       "bt_maxkey", bstp->bt_maxkey);
  557. set_int_field(jnienv, dbclass, retval,
  558.       "bt_minkey", bstp->bt_minkey);
  559. set_int_field(jnienv, dbclass, retval,
  560.       "bt_re_len", bstp->bt_re_len);
  561. set_int_field(jnienv, dbclass, retval,
  562.       "bt_re_pad", bstp->bt_re_pad);
  563. set_int_field(jnienv, dbclass, retval,
  564.       "bt_levels", bstp->bt_levels);
  565. set_int_field(jnienv, dbclass, retval,
  566.       "bt_int_pg", bstp->bt_int_pg);
  567. set_int_field(jnienv, dbclass, retval,
  568.       "bt_leaf_pg", bstp->bt_leaf_pg);
  569. set_int_field(jnienv, dbclass, retval,
  570.       "bt_dup_pg", bstp->bt_dup_pg);
  571. set_int_field(jnienv, dbclass, retval,
  572.       "bt_over_pg", bstp->bt_over_pg);
  573. set_int_field(jnienv, dbclass, retval,
  574.       "bt_free", bstp->bt_free);
  575. set_int_field(jnienv, dbclass, retval,
  576.       "bt_int_pgfree", bstp->bt_int_pgfree);
  577. set_int_field(jnienv, dbclass, retval,
  578.       "bt_leaf_pgfree", bstp->bt_leaf_pgfree);
  579. set_int_field(jnienv, dbclass, retval,
  580.       "bt_dup_pgfree", bstp->bt_dup_pgfree);
  581. set_int_field(jnienv, dbclass, retval,
  582.       "bt_over_pgfree", bstp->bt_over_pgfree);
  583. break;
  584. /* Hash stat structure */
  585. case DB_HASH:
  586. hstp = (DB_HASH_STAT *)statp;
  587. retval = create_default_object(jnienv,
  588.        name_DB_HASH_STAT);
  589. dbclass = get_class(jnienv, name_DB_HASH_STAT);
  590. /* Set the individual fields */
  591. set_int_field(jnienv, dbclass, retval,
  592.       "hash_magic", hstp->hash_magic);
  593. set_int_field(jnienv, dbclass, retval,
  594.       "hash_version", hstp->hash_version);
  595. set_int_field(jnienv, dbclass, retval,
  596.       "hash_metaflags", hstp->hash_metaflags);
  597. set_int_field(jnienv, dbclass, retval,
  598.       "hash_nkeys", hstp->hash_nkeys);
  599. set_int_field(jnienv, dbclass, retval,
  600.       "hash_ndata", hstp->hash_ndata);
  601. set_int_field(jnienv, dbclass, retval,
  602.       "hash_pagesize", hstp->hash_pagesize);
  603. set_int_field(jnienv, dbclass, retval,
  604.       "hash_nelem", hstp->hash_nelem);
  605. set_int_field(jnienv, dbclass, retval,
  606.       "hash_ffactor", hstp->hash_ffactor);
  607. set_int_field(jnienv, dbclass, retval,
  608.       "hash_buckets", hstp->hash_buckets);
  609. set_int_field(jnienv, dbclass, retval,
  610.       "hash_free", hstp->hash_free);
  611. set_int_field(jnienv, dbclass, retval,
  612.       "hash_bfree", hstp->hash_bfree);
  613. set_int_field(jnienv, dbclass, retval,
  614.       "hash_bigpages", hstp->hash_bigpages);
  615. set_int_field(jnienv, dbclass, retval,
  616.       "hash_big_bfree", hstp->hash_big_bfree);
  617. set_int_field(jnienv, dbclass, retval,
  618.       "hash_overflows", hstp->hash_overflows);
  619. set_int_field(jnienv, dbclass, retval,
  620.       "hash_ovfl_free", hstp->hash_ovfl_free);
  621. set_int_field(jnienv, dbclass, retval,
  622.       "hash_dup", hstp->hash_dup);
  623. set_int_field(jnienv, dbclass, retval,
  624.       "hash_dup_free", hstp->hash_dup_free);
  625. break;
  626. case DB_QUEUE:
  627. qstp = (DB_QUEUE_STAT *)statp;
  628. retval = create_default_object(jnienv,
  629.        name_DB_QUEUE_STAT);
  630. dbclass = get_class(jnienv, name_DB_QUEUE_STAT);
  631. /* Set the individual fields */
  632. set_int_field(jnienv, dbclass, retval,
  633.       "qs_magic", qstp->qs_magic);
  634. set_int_field(jnienv, dbclass, retval,
  635.       "qs_version", qstp->qs_version);
  636. set_int_field(jnienv, dbclass, retval,
  637.       "qs_metaflags", qstp->qs_metaflags);
  638. set_int_field(jnienv, dbclass, retval,
  639.       "qs_nkeys", qstp->qs_nkeys);
  640. set_int_field(jnienv, dbclass, retval,
  641.       "qs_ndata", qstp->qs_ndata);
  642. set_int_field(jnienv, dbclass, retval,
  643.       "qs_pagesize", qstp->qs_pagesize);
  644. set_int_field(jnienv, dbclass, retval,
  645.       "qs_pages", qstp->qs_pages);
  646. set_int_field(jnienv, dbclass, retval,
  647.       "qs_re_len", qstp->qs_re_len);
  648. set_int_field(jnienv, dbclass, retval,
  649.       "qs_re_pad", qstp->qs_re_pad);
  650. set_int_field(jnienv, dbclass, retval,
  651.       "qs_pgfree", qstp->qs_pgfree);
  652. set_int_field(jnienv, dbclass, retval,
  653.       "qs_first_recno", qstp->qs_first_recno);
  654. set_int_field(jnienv, dbclass, retval,
  655.       "qs_cur_recno", qstp->qs_cur_recno);
  656. break;
  657. /* That's all the database types we're aware of! */
  658. default:
  659. report_exception(jnienv,
  660.  "Db.stat not implemented for types"
  661.  "other than HASH, BTREE and RECNO",
  662.  EINVAL, 0);
  663. break;
  664. }
  665. free(statp);
  666. }
  667. JAVADB_API_END(db);
  668. return (retval);
  669. }
  670. JNIEXPORT jint JNICALL Java_com_sleepycat_db_Db_sync
  671.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jint flags)
  672. {
  673. int err;
  674. DB *db = get_DB(jnienv, jthis);
  675. if (!verify_non_null(jnienv, db))
  676. return (0);
  677. JAVADB_API_BEGIN(db, jthis);
  678. err = db->sync(db, flags);
  679. if (err != DB_INCOMPLETE)
  680. verify_return(jnienv, err, 0);
  681. JAVADB_API_END(db);
  682. return (err);
  683. }
  684. JNIEXPORT jboolean JNICALL Java_com_sleepycat_db_Db_get_1byteswapped
  685.   (JNIEnv *jnienv, /*Db*/ jobject jthis)
  686. {
  687. DB *db;
  688. jboolean retval;
  689. db = get_DB(jnienv, jthis);
  690. if (!verify_non_null(jnienv, db))
  691. return (0);
  692. JAVADB_API_BEGIN(db, jthis);
  693. retval = db->get_byteswapped(db) ? 1 : 0;
  694. JAVADB_API_END(db);
  695. return (retval);
  696. }
  697. JNIEXPORT jint JNICALL Java_com_sleepycat_db_Db_get_1type
  698.   (JNIEnv *jnienv, /*Db*/ jobject jthis)
  699. {
  700. DB *db;
  701. db = get_DB(jnienv, jthis);
  702. if (!verify_non_null(jnienv, db))
  703. return (0);
  704. return ((jint)db->type);
  705. }
  706. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db__1open
  707.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jstring file, jstring database,
  708.    jint type, jint flags, jint mode)
  709. {
  710. int err;
  711. DB *db;
  712. JSTR dbfile;
  713. JSTR dbdatabase;
  714. /* Java is assumed to be threaded. */
  715. flags |= DB_THREAD;
  716. db = get_DB(jnienv, jthis);
  717. if (jstr_lock(&dbfile, jnienv, file) != 0)
  718. goto out2;
  719. if (jstr_lock(&dbdatabase, jnienv, database) != 0)
  720. goto out1;
  721. if (verify_non_null(jnienv, db)) {
  722. JAVADB_API_BEGIN(db, jthis);
  723. err = db->open(db, dbfile.string, dbdatabase.string,
  724.        (DBTYPE)type, flags, mode);
  725. verify_return(jnienv, err, EXCEPTION_FILE_NOT_FOUND);
  726. JAVADB_API_END(db);
  727. }
  728.  out1:
  729. jstr_unlock(&dbdatabase, jnienv);
  730.  out2:
  731. jstr_unlock(&dbfile, jnienv);
  732. }
  733. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_upgrade
  734.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jstring name,
  735.    jint flags)
  736. {
  737. int err;
  738. DB *db = get_DB(jnienv, jthis);
  739. JSTR j_name;
  740. if (verify_non_null(jnienv, db)) {
  741. JAVADB_API_BEGIN(db, jthis);
  742. if (jstr_lock(&j_name, jnienv, name) != 0)
  743. goto out;
  744. err = db->upgrade(db, j_name.string, flags);
  745. verify_return(jnienv, err, 0);
  746. JAVADB_API_END(db);
  747. }
  748.  out:
  749. jstr_unlock(&j_name, jnienv);
  750. }
  751. static int java_verify_callback(void *handle, const void *str_arg)
  752. {
  753. char *str;
  754. struct verify_callback_struct *vc;
  755. int len;
  756. jthrowable except;
  757. JNIEnv *jnienv;
  758. str = (char *)str_arg;
  759. vc = (struct verify_callback_struct *)handle;
  760. jnienv = vc->env;
  761. len = strlen(str)+1;
  762. if (len > vc->nbytes) {
  763. vc->nbytes = len;
  764. vc->bytes = (*jnienv)->NewByteArray(jnienv, len);
  765. }
  766. (*jnienv)->SetByteArrayRegion(jnienv, vc->bytes, 0, len, (jbyte*)str);
  767. (*jnienv)->CallVoidMethod(jnienv, vc->streamobj,
  768.   vc->writemid, vc->bytes, 0, len-1);
  769. if ((except = (*jnienv)->ExceptionOccurred(jnienv)) != NULL)
  770. return (EIO);
  771. return (0);
  772. }
  773. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db_verify
  774.   (JNIEnv *jnienv, /*Db*/ jobject jthis, jstring name,
  775.    jstring subdb, jobject stream, jint flags)
  776. {
  777. int err;
  778. DB *db;
  779. JSTR j_name;
  780. JSTR j_subdb;
  781. struct verify_callback_struct vcs;
  782. jclass streamclass;
  783. db = get_DB(jnienv, jthis);
  784. if (!verify_non_null(jnienv, db))
  785. return;
  786. JAVADB_API_BEGIN(db, jthis);
  787. if (jstr_lock(&j_name, jnienv, name) != 0)
  788. goto out2;
  789. if (jstr_lock(&j_subdb, jnienv, subdb) != 0)
  790. goto out1;
  791. /* set up everything we need for the callbacks */
  792. vcs.env = jnienv;
  793. vcs.streamobj = stream;
  794. vcs.nbytes = 100;
  795. vcs.bytes = (*jnienv)->NewByteArray(jnienv, vcs.nbytes);
  796. /* get the method ID for OutputStream.write(byte[], int, int); */
  797. streamclass = (*jnienv)->FindClass(jnienv, "java/io/OutputStream");
  798. vcs.writemid = (*jnienv)->GetMethodID(jnienv, streamclass,
  799.       "write", "([BII)V");
  800. /* invoke verify - this will invoke the callback repeatedly. */
  801. err = __db_verify_internal(db, j_name.string, j_subdb.string,
  802.    &vcs, java_verify_callback, flags);
  803. verify_return(jnienv, err, 0);
  804. out1:
  805. jstr_unlock(&j_subdb, jnienv);
  806. out2:
  807. jstr_unlock(&j_name, jnienv);
  808. JAVADB_API_END(db);
  809. }
  810. JNIEXPORT void JNICALL Java_com_sleepycat_db_Db__1finalize
  811.     (JNIEnv *jnienv, jobject jthis,
  812.      jobject /*DbErrcall*/ errcall, jstring errpfx)
  813. {
  814. DB_JAVAINFO *dbinfo;
  815. DB *db;
  816. dbinfo = get_DB_JAVAINFO(jnienv, jthis);
  817. db = get_DB(jnienv, jthis);
  818. DB_ASSERT(dbinfo != NULL);
  819. /* Note: We can never be sure if the underlying DB is attached to
  820.  * a DB_ENV that was already closed.  Sure, that's a user error,
  821.  * but it shouldn't crash the VM.  Therefore, we cannot just
  822.  * automatically close if the handle indicates we are not yet
  823.  * closed.  The best we can do is detect this and report it.
  824.  */
  825. if (db != NULL) {
  826. /* If this error occurs, this object was never closed. */
  827. report_errcall(jnienv, errcall, errpfx,
  828.        "Db.finalize: open Db object destroyed");
  829. }
  830. /* Shouldn't see this object again, but just in case */
  831. set_private_dbobj(jnienv, name_DB, jthis, 0);
  832. set_private_info(jnienv, name_DB, jthis, 0);
  833. dbji_destroy(dbinfo, jnienv);
  834. }