txn_rec.c
上传用户:tsgydb
上传日期:2007-04-14
资源大小:10674k
文件大小:9k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /*-
  2.  * See the file LICENSE for redistribution information.
  3.  *
  4.  * Copyright (c) 1996, 1997, 1998, 1999, 2000
  5.  * Sleepycat Software.  All rights reserved.
  6.  */
  7. /*
  8.  * Copyright (c) 1996
  9.  * The President and Fellows of Harvard University.  All rights reserved.
  10.  *
  11.  * Redistribution and use in source and binary forms, with or without
  12.  * modification, are permitted provided that the following conditions
  13.  * are met:
  14.  * 1. Redistributions of source code must retain the above copyright
  15.  *    notice, this list of conditions and the following disclaimer.
  16.  * 2. Redistributions in binary form must reproduce the above copyright
  17.  *    notice, this list of conditions and the following disclaimer in the
  18.  *    documentation and/or other materials provided with the distribution.
  19.  * 3. Neither the name of the University nor the names of its contributors
  20.  *    may be used to endorse or promote products derived from this software
  21.  *    without specific prior written permission.
  22.  *
  23.  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  24.  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  25.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  26.  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  27.  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  28.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  29.  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  30.  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  31.  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  32.  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  33.  * SUCH DAMAGE.
  34.  */
  35. #include "db_config.h"
  36. #ifndef lint
  37. static const char revid[] = "$Id: txn_rec.c,v 11.15 2001/01/11 18:19:55 bostic Exp $";
  38. #endif /* not lint */
  39. #ifndef NO_SYSTEM_INCLUDES
  40. #include <sys/types.h>
  41. #endif
  42. #include "db_int.h"
  43. #include "db_page.h"
  44. #include "txn.h"
  45. #include "db_am.h"
  46. #include "db_dispatch.h"
  47. #include "log.h"
  48. #include "common_ext.h"
  49. static int __txn_restore_txn __P((DB_ENV *, DB_LSN *, __txn_xa_regop_args *));
  50. #define IS_XA_TXN(R) (R->xid.size != 0)
  51. /*
  52.  * PUBLIC: int __txn_regop_recover
  53.  * PUBLIC:    __P((DB_ENV *, DBT *, DB_LSN *, db_recops, void *));
  54.  *
  55.  * These records are only ever written for commits.  Normally, we redo any
  56.  * committed transaction, however if we are doing recovery to a timestamp, then
  57.  * we may treat transactions that commited after the timestamp as aborted.
  58.  */
  59. int
  60. __txn_regop_recover(dbenv, dbtp, lsnp, op, info)
  61. DB_ENV *dbenv;
  62. DBT *dbtp;
  63. DB_LSN *lsnp;
  64. db_recops op;
  65. void *info;
  66. {
  67. __txn_regop_args *argp;
  68. int ret;
  69. #ifdef DEBUG_RECOVER
  70. (void)__txn_regop_print(dbenv, dbtp, lsnp, op, info);
  71. #endif
  72. if ((ret = __txn_regop_read(dbenv, dbtp->data, &argp)) != 0)
  73. return (ret);
  74. if (argp->opcode != TXN_COMMIT) {
  75. ret = EINVAL;
  76. goto err;
  77. }
  78. if (op == DB_TXN_FORWARD_ROLL)
  79. ret = __db_txnlist_remove(info, argp->txnid->txnid);
  80. else if (dbenv->tx_timestamp == 0 ||
  81.     argp->timestamp <= (int32_t)dbenv->tx_timestamp)
  82. /*
  83.  * We know this is the backward roll case because we
  84.  * are never called during ABORT or OPENFILES.
  85.  */
  86. ret = __db_txnlist_add(dbenv, info, argp->txnid->txnid, 0);
  87. else
  88. /*
  89.  * This is commit record, but we failed the timestamp check
  90.  * so we should treat it as an abort and add it to the list
  91.  * as an aborted record.
  92.  */
  93. ret = __db_txnlist_add(dbenv, info, argp->txnid->txnid, 1);
  94. if (ret == 0)
  95. *lsnp = argp->prev_lsn;
  96. err: __os_free(argp, 0);
  97. return (ret);
  98. }
  99. /*
  100.  * PUBLIC: int __txn_xa_regop_recover
  101.  * PUBLIC:    __P((DB_ENV *, DBT *, DB_LSN *, db_recops, void *));
  102.  *
  103.  * These records are only ever written for prepares.
  104.  */
  105. int
  106. __txn_xa_regop_recover(dbenv, dbtp, lsnp, op, info)
  107. DB_ENV *dbenv;
  108. DBT *dbtp;
  109. DB_LSN *lsnp;
  110. db_recops op;
  111. void *info;
  112. {
  113. __txn_xa_regop_args *argp;
  114. int ret;
  115. #ifdef DEBUG_RECOVER
  116. (void)__txn_xa_regop_print(dbenv, dbtp, lsnp, op, info);
  117. #endif
  118. if ((ret = __txn_xa_regop_read(dbenv, dbtp->data, &argp)) != 0)
  119. return (ret);
  120. if (argp->opcode != TXN_PREPARE) {
  121. ret = EINVAL;
  122. goto err;
  123. }
  124. ret = __db_txnlist_find(info, argp->txnid->txnid);
  125. /*
  126.  * If we are rolling forward, then an aborted prepare
  127.  * indicates that this is the last record we'll see for
  128.  * this transaction ID and we should remove it from the
  129.  * list.
  130.  */
  131. if (op == DB_TXN_FORWARD_ROLL && ret == 1)
  132. ret = __db_txnlist_remove(info, argp->txnid->txnid);
  133. else if (op == DB_TXN_BACKWARD_ROLL && ret != 0) {
  134. /*
  135.  * On the backward pass, we have three possibilities:
  136.  * 1. The transaction is already committed, no-op.
  137.  * 2. The transaction is not committed and we are XA, treat
  138.  *  like commited and roll forward so that can be committed
  139.  *  or aborted late.
  140.  * 3. The transaction is not committed and we are not XA
  141.  * mark the transaction as aborted.
  142.  *
  143.  * Cases 2 and 3 are handled here.
  144.  */
  145. /*
  146.  * Should never have seen this transaction unless it was
  147.  * commited.
  148.  */
  149. DB_ASSERT(ret == DB_NOTFOUND);
  150. if (IS_XA_TXN(argp)) {
  151. /*
  152.  * This is an XA prepared, but not yet committed
  153.  * transaction.  We need to add it to the
  154.  * transaction list, so that it gets rolled
  155.  * forward. We also have to add it to the region's
  156.  * internal state so it can be properly aborted
  157.  * or recovered.
  158.  */
  159. if ((ret = __db_txnlist_add(dbenv,
  160.     info, argp->txnid->txnid, 0)) == 0)
  161. ret = __txn_restore_txn(dbenv, lsnp, argp);
  162. } else
  163. ret = __db_txnlist_add(dbenv,
  164.     info, argp->txnid->txnid, 1);
  165. } else
  166. ret = 0;
  167. if (ret == 0)
  168. *lsnp = argp->prev_lsn;
  169. err: __os_free(argp, 0);
  170. return (ret);
  171. }
  172. /*
  173.  * PUBLIC: int __txn_ckp_recover
  174.  * PUBLIC: __P((DB_ENV *, DBT *, DB_LSN *, db_recops, void *));
  175.  */
  176. int
  177. __txn_ckp_recover(dbenv, dbtp, lsnp, op, info)
  178. DB_ENV *dbenv;
  179. DBT *dbtp;
  180. DB_LSN *lsnp;
  181. db_recops op;
  182. void *info;
  183. {
  184. __txn_ckp_args *argp;
  185. int ret;
  186. #ifdef DEBUG_RECOVER
  187. __txn_ckp_print(dbenv, dbtp, lsnp, op, info);
  188. #endif
  189. COMPQUIET(dbenv, NULL);
  190. if ((ret = __txn_ckp_read(dbenv, dbtp->data, &argp)) != 0)
  191. return (ret);
  192. /*
  193.  * Check for 'restart' checkpoint record.  This occurs when the
  194.  * checkpoint lsn is equal to the lsn of the checkpoint record
  195.  * and means that we could set the transaction ID back to 1, so
  196.  * that we don't exhaust the transaction ID name space.
  197.  */
  198. if (argp->ckp_lsn.file == lsnp->file &&
  199.     argp->ckp_lsn.offset == lsnp->offset)
  200. __db_txnlist_gen(info, DB_REDO(op) ? -1 : 1);
  201. *lsnp = argp->last_ckp;
  202. __os_free(argp, 0);
  203. return (DB_TXN_CKP);
  204. }
  205. /*
  206.  * __txn_child_recover
  207.  * Recover a commit record for a child transaction.
  208.  *
  209.  * PUBLIC: int __txn_child_recover
  210.  * PUBLIC:    __P((DB_ENV *, DBT *, DB_LSN *, db_recops, void *));
  211.  */
  212. int
  213. __txn_child_recover(dbenv, dbtp, lsnp, op, info)
  214. DB_ENV *dbenv;
  215. DBT *dbtp;
  216. DB_LSN *lsnp;
  217. db_recops op;
  218. void *info;
  219. {
  220. __txn_child_args *argp;
  221. int ret;
  222. #ifdef DEBUG_RECOVER
  223. (void)__txn_child_print(dbenv, dbtp, lsnp, op, info);
  224. #endif
  225. if ((ret = __txn_child_read(dbenv, dbtp->data, &argp)) != 0)
  226. return (ret);
  227. /*
  228.  * This is a record in a PARENT's log trail indicating that a
  229.  * child commited.  If we are aborting, we need to update the
  230.  * parent's LSN array.  If we are in recovery, then if the
  231.  * parent is commiting, we set ourselves up to commit, else
  232.  * we do nothing.
  233.  */
  234. if (op == DB_TXN_ABORT) {
  235. /* Note that __db_txnlist_lsnadd rewrites its LSN
  236.  * parameter, so you cannot reuse the argp->c_lsn field.
  237.  */
  238. ret = __db_txnlist_lsnadd(dbenv,
  239.     info, &argp->c_lsn, TXNLIST_NEW);
  240. } else if (op == DB_TXN_BACKWARD_ROLL) {
  241. if (__db_txnlist_find(info, argp->txnid->txnid) == 0)
  242. ret = __db_txnlist_add(dbenv, info, argp->child, 0);
  243. else
  244. ret = __db_txnlist_add(dbenv, info, argp->child, 1);
  245. } else
  246. ret = __db_txnlist_remove(info, argp->child);
  247. if (ret == 0)
  248. *lsnp = argp->prev_lsn;
  249. __os_free(argp, 0);
  250. return (ret);
  251. }
  252. /*
  253.  * __txn_restore_txn --
  254.  * Using only during XA recovery.  If we find any transactions that are
  255.  * prepared, but not yet committed, then we need to restore the transaction's
  256.  * state into the shared region, because the TM is going to issue a txn_abort
  257.  * or txn_commit and we need to respond correctly.
  258.  *
  259.  * lsnp is the LSN of the returned LSN
  260.  * argp is the perpare record (in an appropriate structure)
  261.  */
  262. static int
  263. __txn_restore_txn(dbenv, lsnp, argp)
  264. DB_ENV *dbenv;
  265. DB_LSN *lsnp;
  266. __txn_xa_regop_args *argp;
  267. {
  268. DB_TXNMGR *mgr;
  269. TXN_DETAIL *td;
  270. DB_TXNREGION *region;
  271. int ret;
  272. if (argp->xid.size == 0)
  273. return (0);
  274. mgr = dbenv->tx_handle;
  275. region = mgr->reginfo.primary;
  276. R_LOCK(dbenv, &mgr->reginfo);
  277. /* Allocate a new transaction detail structure. */
  278. if ((ret =
  279.     __db_shalloc(mgr->reginfo.addr, sizeof(TXN_DETAIL), 0, &td)) != 0)
  280. return (ret);
  281. /* Place transaction on active transaction list. */
  282. SH_TAILQ_INSERT_HEAD(&region->active_txn, td, links, __txn_detail);
  283. td->txnid = argp->txnid->txnid;
  284. td->begin_lsn = argp->begin_lsn;
  285. td->last_lsn = *lsnp;
  286. td->parent = 0;
  287. td->status = TXN_PREPARED;
  288. td->xa_status = TXN_XA_PREPARED;
  289. memcpy(td->xid, argp->xid.data, argp->xid.size);
  290. td->bqual = argp->bqual;
  291. td->gtrid = argp->gtrid;
  292. td->format = argp->formatID;
  293. R_UNLOCK(dbenv, &mgr->reginfo);
  294. return (0);
  295. }