mirror of
				git://git.code.sf.net/p/cdesktopenv/code
				synced 2025-03-09 15:50:02 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			485 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			485 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * CDE - Common Desktop Environment
 | |
|  *
 | |
|  * Copyright (c) 1993-2012, The Open Group. All rights reserved.
 | |
|  *
 | |
|  * These libraries and programs are free software; you can
 | |
|  * redistribute them and/or modify them under the terms of the GNU
 | |
|  * Lesser General Public License as published by the Free Software
 | |
|  * Foundation; either version 2 of the License, or (at your option)
 | |
|  * any later version.
 | |
|  *
 | |
|  * These libraries and programs are distributed in the hope that
 | |
|  * they will be useful, but WITHOUT ANY WARRANTY; without even the
 | |
|  * implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
 | |
|  * PURPOSE. See the GNU Lesser General Public License for more
 | |
|  * details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public
 | |
|  * License along with these librararies and programs; if not, write
 | |
|  * to the Free Software Foundation, Inc., 51 Franklin Street, Fifth
 | |
|  * Floor, Boston, MA 02110-1301 USA
 | |
|  */
 | |
| /*%%  (c) Copyright 1993, 1994 Hewlett-Packard Company			 */
 | |
| /*%%  (c) Copyright 1993, 1994 International Business Machines Corp.	 */
 | |
| /*%%  (c) Copyright 1993, 1994 Sun Microsystems, Inc.			 */
 | |
| /*%%  (c) Copyright 1993, 1994 Novell, Inc. 				 */
 | |
| /*%%  $TOG: isrepair.c /main/5 1998/04/10 08:04:42 mgreess $ 			 				 */
 | |
| /* @(#)isrepair.c	1.7 93/07/30
 | |
|  * Copyright (c) 1988, 1993 by Sun Microsystems, Inc.
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * isrepair.c
 | |
|  *
 | |
|  * Description:
 | |
|  *	Repair an ISAM file. 
 | |
|  */
 | |
| 
 | |
| 
 | |
| #include "isam_impl.h"
 | |
| #include <stdlib.h>
 | |
| #include <signal.h>
 | |
| #include <stdio.h>
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * err =  isrepair(isfname, verbose)
 | |
|  *
 | |
|  * isrepair repairs an ISAM file. 
 | |
|  *
 | |
|  * The algorithm used is as the following:
 | |
|  *   1. Read the control page of the possibly damaged file. We assume
 | |
|  *	that the control page is not damaged.
 | |
|  *   2. Open a new ISAM file with ~ suffix.
 | |
|  *   3. Scan .rec file (and .var file if variable length records file)
 | |
|  *	retrieve all records not marked as deleted, and write them
 | |
|  *	to the ~ ISAM file.
 | |
|  *   4. Delete the old ISAM file.
 | |
|  *   5. Rename ~ file to the original file name.
 | |
|  *   6. Build all indexes.
 | |
|  *
 | |
|  *   verbose option (if set to nonzero) will print messages to stdout.
 | |
|  */
 | |
| 
 | |
| Static	char *rp_readrecord_v(), *rp_readrecord_f();
 | |
| Static int printkey(int, struct keydesc *, int (*)(const char *, ...));
 | |
| Static void cmd_error(const char *, int (*)(const char *, ...));
 | |
| Static int typeletter();
 | |
| Static rp_readcntlpg();
 | |
| static int  isnoprintf(const char *, ...);
 | |
| 
 | |
| isrepair(isfname, verbose)
 | |
|     char	*isfname;
 | |
|     int		verbose;
 | |
| {
 | |
|   extern      char *rp_readrecord_v(), *rp_readrecord_f();
 | |
|   char	      cntlpg[ISCNTLSIZE];
 | |
|   int	      datfd = -1, indfd = -1, varfd = -1;
 | |
|   int	      minreclen, maxreclen;
 | |
|   int	      nrecords_fromcntl;
 | |
|   int	      varflag;
 | |
|   char	      nameBuf [MAXPATHLEN];
 | |
|   char	      *namebuf;
 | |
|   int	      isfd = -1;
 | |
|   struct keydesc2	keydesc2;
 | |
|   int	      i;
 | |
|   long	      offset, recfile_end;
 | |
|   char	      *prec;
 | |
|   long	      recnum;
 | |
|   int	      nrecords_found, diff;
 | |
|   long        lastrecno;
 | |
|   struct keydesc keydesc;
 | |
|   int	      (*print)(const char *, ...);
 | |
|   sigset_t    oldmask;
 | |
|   sigset_t    allsignals;
 | |
|   char	      Buffer[BUFSIZ];
 | |
|   char	      *buffer = NULL;
 | |
| 
 | |
| 
 | |
|   print = verbose ? printf : isnoprintf;
 | |
|   datfd = indfd = varfd = -1;
 | |
| 
 | |
|   /*
 | |
|    * Open UNIX files.
 | |
|    */
 | |
|   if (strlen(isfname) + 10 >= MAXPATHLEN)
 | |
|     namebuf = (char*) malloc(strlen(isfname) + 10);
 | |
|   else
 | |
|     namebuf = nameBuf;
 | |
| 
 | |
|   (void)strcpy(namebuf, isfname);
 | |
|   _makedat_isfname(namebuf);
 | |
|   datfd = open(namebuf, O_RDONLY);
 | |
|   if (datfd > -1) {
 | |
|     if(fcntl(datfd, F_SETFD, 1) == -1) {
 | |
|       close(datfd);
 | |
|       datfd = -1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   (void)strcpy(namebuf, isfname);
 | |
|   _makeind_isfname(namebuf);
 | |
|   indfd = open(namebuf, O_RDONLY);
 | |
|   if (indfd > -1) {
 | |
|     if(fcntl(indfd, F_SETFD, 1) == -1) {
 | |
|       close(indfd);
 | |
|       indfd = -1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   (void)strcpy(namebuf, isfname);
 | |
|   _makevar_isfname(namebuf);
 | |
|   varfd = open(namebuf, O_RDONLY);
 | |
|   if (varfd > -1) {
 | |
|     if(fcntl(varfd, F_SETFD, 1) == -1) {
 | |
|        close(varfd);
 | |
|        varfd = -1;         
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   (void)print("Reading control page from %s.rec file...\n",
 | |
| 	      isfname);
 | |
|   if (rp_readcntlpg(datfd, cntlpg) == ISERROR) {
 | |
|     (void)print("Cannot read the control page\n");
 | |
|     goto ERROR;
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|    * Check magic. Repair only ISAM files!!!
 | |
|    */
 | |
| 
 | |
|   if (strncmp(cntlpg + CP_MAGIC_OFF, ISMAGIC, strlen(ISMAGIC)) != 0) {
 | |
|     (void)print("Bad magic in %s.rec\n", isfname);
 | |
|     goto ERROR;
 | |
|   }
 | |
| 
 | |
|   varflag = ldint(cntlpg + CP_VARFLAG_OFF);
 | |
|   minreclen = ldint(cntlpg + CP_MINRECLEN_OFF);
 | |
|   maxreclen = ldint(cntlpg + CP_MAXRECLEN_OFF);	
 | |
| 
 | |
|   /*
 | |
|    *  Check for maxreclen field value of -1. This could have occured due to
 | |
|    *  ISMAXRECLEN being incorrectly set to 65535 in an earlier version. If
 | |
|    *  this field is -1 and it's a variable length record, reset to the new
 | |
|    *  value of MAXRECLEN. This means that this field will be repaired when
 | |
|    *  the control block is written back to disk.
 | |
|    */
 | |
| 
 | |
|   if (maxreclen == -1 && varflag) {
 | |
|     maxreclen = ISMAXRECLEN;
 | |
|   }
 | |
| 
 | |
|   lastrecno = ldlong(cntlpg + CP_LASTRECNO_OFF);	
 | |
|   nrecords_fromcntl = ldlong(cntlpg + CP_NRECORDS_OFF);
 | |
| 
 | |
|   /*
 | |
|    * Open output file. Use ~ as suffix.
 | |
|    */
 | |
|   (void)sprintf(namebuf, "%s~", isfname);
 | |
|   (void)print("Opening temporary ISAM file '%s'...\n",
 | |
| 	      namebuf);
 | |
|   isreclen = minreclen;
 | |
|   if ((isfd = isbuild(namebuf, maxreclen, nokey, ISINOUT + ISEXCLLOCK +
 | |
| 		      (varflag?ISVARLEN:0))) == ISERROR) {
 | |
|     (void)print("Cannot open temporary ISAM file %s\n",
 | |
| 		namebuf);
 | |
|     if (iserrno == EEXIST)
 | |
|       (void)print("File %s already exists\n", namebuf);
 | |
|     goto ERROR;
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|    * Scan .rec file and read all undeleted records.
 | |
|    */
 | |
|   (void)print("Salvaging records from %s.rec%s file...\n",
 | |
| 	      isfname, varflag?" (and .var file)" : "");
 | |
| 
 | |
|   offset = ISCNTLSIZE;
 | |
|   recfile_end = lseek(datfd, 0L, 2);
 | |
|   recnum = 1;
 | |
|   nrecords_found = 0;
 | |
|     
 | |
|   while (recnum <= lastrecno && offset < recfile_end - minreclen) {
 | |
| 
 | |
|     if (varflag) {
 | |
|       prec = rp_readrecord_v(datfd, varfd, offset, minreclen, maxreclen);	
 | |
|       offset += minreclen + LONGSIZE;
 | |
|     }
 | |
|     else {
 | |
|       prec = rp_readrecord_f(datfd, offset, minreclen);
 | |
|       offset += minreclen + 1;
 | |
|     }
 | |
| 	
 | |
|     if (prec != NULL) {
 | |
|       if (iswrrec(isfd, recnum, prec) == ISERROR) {
 | |
| 	cmd_error("iswrrec", print);
 | |
| 	goto ERROR;
 | |
|       }
 | |
|       nrecords_found++;
 | |
|     }
 | |
|     recnum++;
 | |
|   }
 | |
| 
 | |
|   diff = nrecords_found - nrecords_fromcntl;
 | |
| 
 | |
|   if (diff == 0)
 | |
|     (void)print("All records found - total %d records\n",
 | |
| 		nrecords_found);
 | |
|   else
 | |
|     (void)print("%d records found - %d records %s than in header\n",
 | |
| 		nrecords_found, diff, diff > 0 ?
 | |
| 		"more" : "less");
 | |
| 
 | |
|   /*
 | |
|    * Close all file descriptors.
 | |
|    */
 | |
|   if(datfd != -1) {
 | |
|     close(datfd);
 | |
|     datfd = -1;
 | |
|   }
 | |
|   if(indfd != -1) {
 | |
|     close(indfd);
 | |
|     indfd = -1;
 | |
|   }
 | |
|   if(varfd != -1) {
 | |
|     close(varfd);
 | |
|     varfd = -1;
 | |
|   }
 | |
|   (void)isclose(isfd);
 | |
| 
 | |
|   (void) sigfillset(&allsignals);
 | |
|   (void) sigprocmask(SIG_SETMASK, &allsignals, &oldmask);
 | |
| 
 | |
|   (void)print("Erasing ISAM file '%s'...\n", isfname);
 | |
|   /*    if (iserase(isfname) != ISOK) {
 | |
| 	cmd_error("iserase", print);
 | |
| 	goto ERROR;
 | |
| 	}
 | |
| 	*/
 | |
|   if (strlen(isfname) + 5 >= MAXPATHLEN)
 | |
|     buffer = (char*) malloc(strlen(isfname) + 5);
 | |
|   else
 | |
|     buffer = Buffer;
 | |
| 
 | |
|   (void)sprintf(buffer,"%s.rec", isfname);
 | |
|   (void)unlink(buffer);
 | |
|   (void)sprintf(buffer,"%s.ind", isfname);
 | |
|   (void)unlink(buffer);
 | |
|   (void)sprintf(buffer,"%s.var", isfname);
 | |
|   (void)unlink(buffer);
 | |
| 
 | |
|   (void)sprintf(namebuf, "%s~", isfname);
 | |
|   (void)print("Renaming ISAM file '%s' to '%s'...\n",
 | |
| 	      namebuf, isfname);
 | |
|   if (isrename(namebuf, isfname) != ISOK) {
 | |
|     cmd_error("isrename", print);
 | |
|     goto ERROR;
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|    * Re-open the file and add keys.
 | |
|    */
 | |
|   if (ldshort(cntlpg + CP_NKEYS_OFF) > 0) {
 | |
|     (void)print("Adding keys...\n");
 | |
| 
 | |
|     if ((isfd = isopen(isfname, ISEXCLLOCK + ISINOUT +
 | |
| 		       (varflag?ISVARLEN:0))) == ISERROR) {
 | |
|       cmd_error("isopen", print);
 | |
|       goto ERROR;
 | |
|     }
 | |
| 	
 | |
|     for (i = 0; i < ldshort(cntlpg + CP_NKEYS_OFF); i++) {
 | |
|       ldkey(&keydesc2, cntlpg + CP_KEYS_OFF + i * K2_LEN);
 | |
|       _iskey_itox(&keydesc2, &keydesc);
 | |
| 
 | |
|       if (keydesc.k_nparts == 0) /* special case for no primary */
 | |
| 	continue;
 | |
| 
 | |
|       printkey (i+1, &keydesc, print);
 | |
| 
 | |
|       if (i == 0) {
 | |
| 	if (isaddprimary(isfd, &keydesc) == ISERROR) {
 | |
| 	  cmd_error("isaddprimary", print);
 | |
| 	  (void)isclose(isfd);
 | |
| 	  goto ERROR;
 | |
| 	}
 | |
|       }
 | |
|       else {
 | |
| 	if (isaddindex(isfd, &keydesc) == ISERROR) {
 | |
| 	  cmd_error("isaddindex", print);
 | |
| 	  (void)isclose(isfd);
 | |
| 	  goto ERROR;
 | |
| 	}
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   (void)isclose(isfd);
 | |
|   (void)sigprocmask(SIG_SETMASK, &oldmask, NULL);
 | |
| 
 | |
|   print("...File repaired\n");
 | |
|   if (buffer != Buffer) free(buffer);
 | |
|   return (ISOK);
 | |
| 
 | |
|  ERROR:
 | |
|   (void)print("\007Didn't repair ISAM file '%s'\n", isfname);
 | |
|   if(datfd != -1) {
 | |
|     close(datfd);
 | |
|   }
 | |
|   if(indfd != -1) {
 | |
|     close(indfd);
 | |
|   }
 | |
|   if(varfd != -1) {
 | |
|     close(varfd);
 | |
|   }
 | |
|   (void)isclose(isfd);
 | |
|   if (namebuf != nameBuf) free(namebuf);
 | |
|   if (buffer != Buffer) free(buffer);
 | |
| 
 | |
|   return (ISERROR);
 | |
| }
 | |
| 
 | |
| /******* low level data access used by the 'repair' utility ******************/
 | |
| 
 | |
| static char	recordbuffer[ISMAXRECLEN + LONGSIZE];
 | |
| 
 | |
| /* rp_readcntlpg() - Read the control page */
 | |
| Static 
 | |
| rp_readcntlpg(datfd, cntlpg)
 | |
|     int		datfd;
 | |
|     char	*cntlpg;
 | |
| {
 | |
|     if (read (datfd, cntlpg, ISCNTLSIZE) != ISCNTLSIZE)
 | |
| 	return (ISERROR);
 | |
| 
 | |
|     return (ISOK);
 | |
| }
 | |
| 
 | |
| /* rp_readrecord_f() - Read a record from .rec file */
 | |
| Static char *
 | |
| rp_readrecord_f(datfd, offset, reclen)
 | |
|     int		datfd;
 | |
|     long	offset;
 | |
|     int		reclen;
 | |
| {
 | |
|     if (lseek(datfd, offset, 0) != offset)
 | |
| 	return ((char *) NULL);
 | |
| 
 | |
|     if (read(datfd, recordbuffer, reclen + 1) != (reclen + 1))
 | |
| 	return ((char *) NULL);
 | |
| 
 | |
|     if (recordbuffer[0] == FL_RECDELETED)
 | |
| 	return ((char *) NULL);
 | |
| 
 | |
|     return (recordbuffer + 1);
 | |
| }
 | |
| 
 | |
| /* rp_readrecord_v() - Read a record from .rec file */
 | |
| Static char *
 | |
| rp_readrecord_v(datfd, varfd, offset, minreclen, maxreclen)
 | |
|     int		datfd, varfd;
 | |
|     long	offset;
 | |
|     int		minreclen, maxreclen;
 | |
| {
 | |
|     long	tailoff;
 | |
|     char	frameheadbuf [2 * SHORTSIZE];
 | |
|     int		taillen;
 | |
|     
 | |
|     if (lseek(datfd, offset, 0) != offset)
 | |
| 	return ((char *) NULL);
 | |
| 
 | |
|     if (read(datfd, recordbuffer, minreclen + LONGSIZE) != (minreclen + LONGSIZE))
 | |
| 	return ((char *) NULL);
 | |
| 
 | |
|     if ((tailoff = ldlong(recordbuffer)) == VL_RECDELETED)
 | |
| 	return ((char *) NULL);
 | |
| 
 | |
|     isreclen = minreclen;
 | |
| 
 | |
|     /* Recover tail of the record */
 | |
|     if (tailoff != VL_RECNOTAIL) {
 | |
| 
 | |
| 	if (lseek(varfd, tailoff, 0) != tailoff)
 | |
| 	    goto OKEXIT;
 | |
| 
 | |
| 	if (read(varfd, frameheadbuf, 2 * SHORTSIZE) != 2 * SHORTSIZE)
 | |
| 	    goto OKEXIT;
 | |
| 
 | |
| 	taillen	 = (int) ldshort(frameheadbuf + VR_TAILLEN_OFF);
 | |
| 
 | |
| 	if (taillen <= 0 || taillen + minreclen > maxreclen)
 | |
| 	    goto OKEXIT;
 | |
| 
 | |
| 	if (read(varfd, recordbuffer + LONGSIZE + isreclen, taillen) != taillen)
 | |
| 	    goto OKEXIT;
 | |
| 
 | |
| 	isreclen += taillen;
 | |
|     }
 | |
| 
 | |
|  OKEXIT:
 | |
|     return (recordbuffer + LONGSIZE);
 | |
| }
 | |
| 
 | |
| static int
 | |
| isnoprintf(const char *s, ...)
 | |
| {
 | |
|     return(0);
 | |
| }
 | |
| 
 | |
| static int
 | |
| printkey(int n, struct keydesc *pkdesc, int (*print)(const char *, ...))
 | |
| {
 | |
|     int		i;
 | |
|     struct keypart *pk;
 | |
| 
 | |
|     if (pkdesc->k_nparts == 0) {
 | |
| 	print("%3d: --- NO PRIMARY KEY ---\n", n);
 | |
| 	return 0;
 | |
|     }
 | |
| 
 | |
|     if (n == 1)
 | |
| 	print("P%3d: %s ", n, (pkdesc->k_flags&ISDUPS) ?
 | |
| 	      "DUPS  " : "NODUPS");
 | |
|     else
 | |
| 	print(" %3d: %s ", n, (pkdesc->k_flags&ISDUPS) ?
 | |
| 	      "DUPS  " : "NODUPS");
 | |
| 
 | |
| 
 | |
|     for (i = 0; i < pkdesc->k_nparts; i++) {
 | |
| 	pk = pkdesc->k_part + i;
 | |
| 	print(" %d%c%d%s", pk->kp_start, 
 | |
| 	       typeletter(pk->kp_type & ~ISDESC), pk->kp_leng,
 | |
| 	       (pk->kp_type & ISDESC)?"D":" ");
 | |
|     }
 | |
|     print("\n");
 | |
|     return 1;
 | |
| }
 | |
| 
 | |
| static void
 | |
| cmd_error(const char *str, int (*print)(const char *, ...))
 | |
| {
 | |
|     (void)print("%s: ISAM error %d\n", str, iserrno);
 | |
| }
 | |
| 
 | |
| static int
 | |
| typeletter(type)
 | |
|     int 	type;
 | |
| {
 | |
|     switch (type) {
 | |
|     case INTTYPE:
 | |
| 	return 'I';
 | |
|     case LONGTYPE:
 | |
| 	return 'L';
 | |
|     case FLOATTYPE:
 | |
| 	return 'F';
 | |
|     case DOUBLETYPE:
 | |
| 	return 'D';
 | |
|     case CHARTYPE:
 | |
| 	return 'C';
 | |
|     case BINTYPE:
 | |
| 	return 'B';
 | |
|     default:
 | |
| 	assert(0);
 | |
|     }
 | |
|     /* NOTREACHED */
 | |
| }
 |