home *** CD-ROM | disk | FTP | other *** search
- /*
- * c-tree Server Lock Routines
- *
- * This program is the CONFIDENTIAL and PROPRIETARY property
- * of FairCom(R) Corporation. Any unauthorized use, reproduction or
- * transfer of this program is strictly prohibited.
- *
- * Copyright (c) 1985, 1986, 1987, 1988, 1989 FairCom Corporation
- * (Subject to limited distribution and
- * restricted disclosure only.)
- * *** ALL RIGHTS RESERVED ***
- *
- * 4006 West Broadway
- * Columbia, MO 65203
- *
- *
- * c-tree(R) Version 4.3
- * Release C
- * February 7, 1989 17:30
- *
- */
-
- #include "ctstdr.h" /* standard i/o header */
- #include "ctoptn.h" /* c-tree configuration options */
- #include "cterrc.h" /* c-tree error codes */
- #include "ctstrc.h" /* c-tree data structures */
- #include "ctgvar.h" /* c-tree global variables */
- #include "ctcomm.h"
-
- #ifdef CTSERVER
-
- extern SSTATUS cts_stat;
-
- #define cts_wl cts_stat.scts_wl
- #define cts_rl cts_stat.scts_rl
- #define cts_ul cts_stat.scts_ul
-
- COUNT uerr();
-
- extern COUNT cur_usrid;
-
- LOCAL RECLOK *poplst()
- {
- FAST RECLOK *retval;
-
- if ((retval = ct_frelst) == (RECLOK *) NULL)
- uerr_cod = SFRE_ERR;
- else
- ct_frelst = ct_frelst->rllink;
- return(retval);
- }
-
- VOID pshlst( plok,ctnum)
- PFAST RECLOK *plok;
- PFAST CTFILE *ctnum;
- {
- RECLOK *prvlst,*nxtlst;
-
- if ((prvlst = plok->rlrlnk) == NULL) {
- if (ctnum->rlokh != plok)
- terr(980);
- } else {
- if (prvlst->rllink != plok)
- terr(981);
- }
- if ((nxtlst = plok->rllink) == NULL) {
- if (ctnum->rlokt != plok)
- terr(982);
- } else {
- if (nxtlst->rlrlnk != plok)
- terr(983);
- }
-
- if (prvlst != NULL)
- prvlst->rllink = nxtlst;
- else
- ctnum->rlokh = nxtlst;
- if (nxtlst != NULL)
- nxtlst->rlrlnk = prvlst;
- else
- ctnum->rlokt = prvlst;
-
- plok->rllink = ct_frelst;
- ct_frelst = plok;
- }
-
- COUNT cts_lok(usrn,filno,loktyp,mode,recbyt)
- COUNT usrn,filno,loktyp,mode;
- POINTER recbyt;
- {
- FAST RECLOK *pp,*cp;
- RECLOK *np;
- FAST CTFILE *ctnum;
-
- ctnum = ct_key + filno;
-
- /* check for empty lock list */
- if ((cp = ctnum->rlokh) == NULL) {
- if (mode == CHK_LOCK)
- return(NO_ERROR);
- if (mode == MST_LOCK)
- return(DADV_ERR);
- if ((cp = poplst()) == NULL)
- return(uerr_cod);
- ctnum->rlokh = ctnum->rlokt = cp;
- cp->rllink = cp->rlrlnk = NULL;
- goto lokret;
- }
-
- /* check for lock past last existing lock */
- if (recbyt > ctnum->rlokt->rlbpos) {
- if (mode == CHK_LOCK)
- return(NO_ERROR);
- if (mode == MST_LOCK)
- return(DADV_ERR);
- if ((cp = poplst()) == NULL)
- return(uerr_cod);
- cp->rlrlnk = ctnum->rlokt;
- cp->rllink = NULL;
- ctnum->rlokt->rllink = cp;
- ctnum->rlokt = cp;
- goto lokret;
- }
-
- /* check for insertion into lock list */
- cp = ctnum->rlokh;
- pp = NULL;
- while (cp != NULL && cp->rlbpos < recbyt) {
- pp = cp;
- cp = cp->rllink;
- }
- if (cp == NULL)
- terr(984);
-
- if (cp->rlbpos == recbyt) {
- if (cp->rlusrn != usrn) {
- if (mode != ADD_LOCK)
- return(DADV_ERR);
- if (cp->rltype == CTWRITE || loktyp == CTWRITE)
- return(DLOK_ERR);
- /* else read lock: check if user further along list */
- np = cp->rllink;
- while (np && np->rlbpos == recbyt) {
- if (np->rlusrn == usrn)
- return(NO_ERROR);
- np = np->rllink;
- }
- /* else drop through to add lock in front of tie */
- } else if (mode != ADD_LOCK) {
- /* position & user agree: check for read lock */
- if (cp->rltype == CTREAD && (mode == MST_LOCK ||
- (cp->rllink != NULL &&
- cp->rllink->rlbpos == recbyt)))
- return(DADV_ERR);
- else
- return(NO_ERROR);
- } else if (cp->rltype == loktyp || (cp->rltype == CTWRITE &&
- loktyp == CTREAD)) {
- cp->rltype = loktyp;
- return(NO_ERROR);
- } else { /* check for other user at same location */
- np = cp->rllink;
- if (np == NULL || np->rlbpos > recbyt) {
- cp->rltype = loktyp;
- return(NO_ERROR);
- } else
- return(DLOK_ERR);
- }
- }
-
- /* if testing lock, check test mode */
- if (mode == CHK_LOCK)
- return(NO_ERROR);
- if (mode == MST_LOCK)
- return(DADV_ERR);
-
- /* cp->rlbpos > recbyt || insert in front of tie */
- np = cp;
- if ((cp = poplst()) == NULL)
- return(uerr_cod);
- if (pp == NULL)
- ctnum->rlokh = cp;
- else
- pp->rllink = cp;
- cp->rlrlnk = pp;
- cp->rllink = np;
- np->rlrlnk = cp;
- /* goto lokret */
-
- lokret:
- cp->rlusrn = usrn;
- cp->rlbpos = recbyt;
- cp->rltype = loktyp;
- return(NO_ERROR);
- }
-
- LOCAL COUNT dellst(usrn,filno,recbyt)
- COUNT usrn,filno;
- POINTER recbyt;
- {
- FAST RECLOK *cp;
- FAST CTFILE *ctnum;
-
- ctnum = ct_key + filno;
- cp = ctnum->rlokh;
-
- while (cp != NULL && cp->rlbpos < recbyt)
- cp = cp->rllink;
- if (cp == NULL || cp->rlbpos != recbyt)
- return(UDLK_ERR);
-
- while (cp != NULL && cp->rlbpos == recbyt && cp->rlusrn != usrn)
- cp = cp->rllink;
- if (cp == NULL || cp->rlbpos != recbyt)
- return(UDLK_ERR);
-
- pshlst(cp,ctnum);
- return(NO_ERROR);
- }
-
- /* --------------------------------------------------------------------
- LOCK index node
- */
-
- COUNT LOCK(node,knum) /* node == 0 => header */
- LONG node;
- PFAST KEYFILE *knum;
- {
- return(NO_ERROR);
- }
-
- /* --------------------------------------------------------------------
- UNLOCK index file node
- */
-
- COUNT UNLOCK(node,knum)
- LONG node;
- PFAST KEYFILE *knum;
- {
- return(NO_ERROR);
- }
-
-
- /* --------------------------------------------------------------------
- write LOCK data record
- */
-
- COUNT DLOCK(recbyt,dnum) /* recbyt == 0 => header record */
- POINTER recbyt;
- PFAST DATFILE *dnum;
- {
-
- /*
- * the data record ct_locks are NOT guaranteed to be deadlock free. Therefore,
- * except for the header record which is only locked by internal c-tree
- * maintenance requests, the data record lock routine returns instead of
- * sleeping when a lock is denied because of a competing lock. The application
- * program must deal with the appropriate action if a lock is denied to a
- * data record.
- */
-
- cts_wl++;
- return(uerr_cod = cts_lok(cur_usrid,dnum->filnum,CTWRITE,
- ADD_LOCK,recbyt));
- }
-
- /* --------------------------------------------------------------------
- read LOCK data record
- */
-
- COUNT RLOCK(recbyt,dnum) /* recbyt == 0 => header record */
- POINTER recbyt;
- PFAST DATFILE *dnum;
- {
- cts_rl++;
- return(uerr_cod = cts_lok(cur_usrid,dnum->filnum,CTREAD,
- ADD_LOCK,recbyt));
- }
-
-
- /* --------------------------------------------------------------------
- UNLOCK data record
- */
-
- COUNT UDLOCK(recbyt,dnum)
- POINTER recbyt;
- PFAST DATFILE *dnum;
- {
- cts_ul++;
- return(uerr_cod = dellst(cur_usrid,dnum->filnum,recbyt));
- }
-
- #endif
-
- /* end of ctslok.c */
-