buflock.c
上传用户:dgyhgb
上传日期:2007-01-07
资源大小:676k
文件大小:6k
源码类别:

SQL Server

开发平台:

Unix_Linux

  1. /*
  2.  *  buflock.c  - This file contains the functions which support main
  3.  *               operations lock/enforce/unlock/tact.
  4.  *               Kernel of GNU SQL-server. Buffer 
  5.  *
  6.  *    The functions deal with three parts of information. Page table
  7.  *  contains pages' descriptors including all information concerning
  8.  *  satisfied locks and presence of pages in the pool of buffers. Hash
  9.  *  table serves a fast search of a page descriptor by the corresponding
  10.  *  page number. Queue table contains all waiting locks.
  11.  *
  12.  * This file is a part of GNU SQL Server
  13.  *
  14.  *  Copyright (c) 1996, 1997, Free Software Foundation, Inc
  15.  *  Developed at the Institute of System Programming
  16.  *  This file is written by  Vera Ponomarenko
  17.  *
  18.  *  This program is free software; you can redistribute it and/or modify
  19.  *  it under the terms of the GNU General Public License as published by
  20.  *  the Free Software Foundation; either version 2 of the License, or
  21.  *  (at your option) any later version.
  22.  *
  23.  *  This program is distributed in the hope that it will be useful,
  24.  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
  25.  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  26.  *  GNU General Public License for more details.
  27.  *
  28.  *  You should have received a copy of the GNU General Public License
  29.  *  along with this program; if not, write to the Free Software
  30.  *  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  31.  *
  32.  *  Contacts:   gss@ispras.ru
  33.  *
  34.  */
  35. /* $Id: buflock.c,v 1.245 1997/03/31 03:46:38 kml Exp $ */
  36. #include "setup_os.h"
  37. #include <assert.h>
  38. #include "inpop.h"
  39. #include "bufdefs.h"
  40. #include "fdeclbuf.h"
  41. /*
  42.  * TIMER SERVICE
  43.  */
  44. u2_t C_tact = 0; /* current tact number */
  45. extern i4_t MAXTACT;
  46. extern struct PAGE *hash_table[];
  47. #define size2b sizeof(u2_t)
  48. void
  49. tactup (void)
  50. {
  51.   if (++C_tact == MAXTACT)
  52.     C_tact = 0;
  53. }
  54. /********************************* next tact ********************************/
  55. void
  56. tact (void)
  57. {
  58.   register i4_t i;
  59.   register struct PAGE *page, *page0;
  60.   tactup ();
  61.   for (i = 0; i < HASHSIZE; i++)
  62.     if ((page = page0 = hash_table[i]) != NULL)
  63.       do
  64. {
  65.   for (; page->p_queue != NULL; page = page->p_next)
  66.             if (page->p_queue->w_tact == C_tact)
  67.               {
  68.                 u2_t trnum;
  69.                 trnum = page->p_queue->w_conn;
  70.                 if (page->p_queue->w_prget == 1)
  71.                   buf_to_user (trnum, BACKUP);
  72.                 else
  73.                   user_p (trnum, BACKUP);
  74.                 out_first (page);
  75.               }
  76.             else
  77.               break;
  78. }
  79.       while (page != page0);
  80. }
  81. /*****************************************************************************
  82.                             INTERFACE FUNCTIONS
  83. ****************************** lock a page **********************************/
  84. int
  85. buflock (u2_t conn, u2_t segn, u2_t pn, char type, u2_t prget)
  86. /* conn - a connection to user */
  87. /* segn, pn - full page number */
  88.       /* type - type of lock */
  89.       /* prdet - get page or not */
  90. {
  91.   struct PAGE *page;
  92.   page = find_page (segn, pn);
  93.   if (page == NULL)
  94.     {
  95.       page = new_page (segn, pn);
  96.       page->p_ltype = type;
  97.     }
  98.   else
  99.     {
  100.       u2_t ptype;
  101.       struct BUFF *buf;
  102.       ptype = page->p_ltype;
  103.       if ((ptype == STRONG) || (type == STRONG && ptype == WEAK))
  104. {
  105.           struct WAIT *wait;
  106.   wait = new_wait (conn, type, C_tact, prget);
  107.   into_wait (page, wait);
  108.           PRINTF (("BUF.buflock.e: ret -1 sn=%d,pn=%d, ptype = %d, "
  109.                    "type = %dn", segn, pn, ptype, type));
  110.   return (-1);
  111. }
  112.       else
  113. page->p_ltype = type;
  114.       if ((buf = page->p_buf) != NULL) /* if it can be locked */
  115. if (buf->b_status < LOCKED)
  116.   change_prio (buf, LOCKED);
  117.     }
  118.   if (page->p_ltype == WEAK)
  119.     page->p_status++;
  120.   return (0);
  121. }
  122. /**************************** enforce a lock ********************************/
  123. int
  124. enforce (u2_t conn, u2_t segn, u2_t pn)
  125. /* conn - connection to user   */
  126. /* segn, pn - full page number */
  127. {
  128.   struct PAGE *page;
  129.   page = find_page (segn, pn);
  130.   if (page == NULL)
  131.     PRINTF (("BUF.enforce: sn=%d,pn=%d, status = %dn",
  132.              segn,pn, page->p_status));
  133.   assert (page != NULL);
  134.   if (page->p_status == 1)
  135.     { /* if lock can be enforced immediately */
  136.       page->p_ltype  = STRONG;
  137.       page->p_status = 0;
  138.       return (0);
  139.     }
  140.   else
  141.     {
  142.       into_wait (page, new_wait (conn, STRONG, C_tact, 0));
  143.       PRINTF (("BUF.enforce.e: ret -1 sn=%d,pn=%dn", segn, pn));
  144.       return (-1);
  145.     }
  146. }
  147. /***************************** unlock a page ********************************/
  148. void
  149. unlock (u2_t segn, u2_t lnum, char *p)
  150. {
  151.   register i4_t stat;
  152.   u2_t i, trnum, prg, pn;
  153.   struct PAGE *page;
  154.   struct WAIT *wait;
  155.   struct BUFF *buf;
  156.   for (i = 0; i < lnum; i++)
  157.     {
  158.       BUFUPACK(p,pn);
  159.       page = find_page (segn, pn);
  160.       if (page == NULL)
  161.         PRINTF (("BUF.unlock: sn=%d,lnum=%d,pn=%d,i=%dn",segn,lnum,pn,i));
  162.       assert (page != NULL);
  163.       switch(page->p_ltype)
  164. {
  165.         case  WEAK:
  166.           stat = (--page->p_status);
  167.           wait = page->p_queue;
  168.   if (wait == NULL)
  169.     {
  170.       if (stat == 0)
  171. page->p_ltype = NO_LOCK;
  172.       if (stat == 0 && (buf = page->p_buf) != NULL)
  173. change_prio (buf, FREE);
  174.     }
  175.   else if (stat==0)
  176.     {/* if somebody wait for a STRONG LOCK and it's possible now */ 
  177.       page->p_ltype = STRONG;
  178.       prg = wait->w_prget;
  179.       trnum = wait->w_conn;
  180.       out_first (page);
  181.       if (prg == 1)
  182. {
  183.   buf = get (segn, pn, 'r');
  184.   buf_to_user (trnum, buf->b_seg->keyseg);
  185. }
  186.       else
  187. user_p (trnum, GOOD);
  188.     }
  189.           /* else we just released one of the weak locks */ 
  190.           break;
  191.         case STRONG:
  192.   if (page->p_queue == NULL)
  193.     {
  194.       page->p_ltype = NO_LOCK;
  195.       if ((buf = page->p_buf) != NULL)
  196.      change_prio (buf, FREE);
  197.     }
  198.   else
  199.     set_weak_locks (page);
  200.           break;
  201.         default:
  202.           assert("incorrect type of lock??");
  203. }
  204.     }
  205. }