375 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			375 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2000 Silicon Graphics, Inc.  All Rights Reserved.
 | |
|  * 
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms of version 2 of the GNU General Public License as
 | |
|  * published by the Free Software Foundation.
 | |
|  * 
 | |
|  * This program is distributed in the hope that it would be useful, but
 | |
|  * WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 | |
|  * 
 | |
|  * Further, this software is distributed without any warranty that it is
 | |
|  * free of the rightful claim of any third person regarding infringement
 | |
|  * or the like.  Any license provided herein, whether implied or
 | |
|  * otherwise, applies only to this software file.  Patent licenses, if
 | |
|  * any, provided herein do not apply to combinations of this program with
 | |
|  * other software, or any other product whatsoever.
 | |
|  * 
 | |
|  * You should have received a copy of the GNU General Public License along
 | |
|  * with this program; if not, write the Free Software Foundation, Inc., 59
 | |
|  * Temple Place - Suite 330, Boston MA 02111-1307, USA.
 | |
|  * 
 | |
|  * Contact information: Silicon Graphics, Inc., 1600 Amphitheatre Pkwy,
 | |
|  * Mountain View, CA  94043, or:
 | |
|  * 
 | |
|  * http://www.sgi.com 
 | |
|  * 
 | |
|  * For further information regarding this notice, see: 
 | |
|  * 
 | |
|  * http://oss.sgi.com/projects/GenInfo/NoticeExplan/
 | |
|  */
 | |
| /************
 | |
| 
 | |
| 64 bits in a Cray word
 | |
| 
 | |
| 				12345678901234567890123456789012
 | |
| 1234567890123456789012345678901234567890123456789012345678901234
 | |
| ________________________________________________________________
 | |
| <    pid       >< word-offset in file (same #) ><    pid       >
 | |
| 
 | |
| 1234567890123456789012345678901234567890123456789012345678901234
 | |
| ________________________________________________________________
 | |
| <    pid       >< offset in file of this word  ><    pid       >
 | |
| 
 | |
| 
 | |
| 8 bits to a bytes == character
 | |
|  NBPW            8 
 | |
| ************/
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <sys/param.h>
 | |
| #ifdef UNIT_TEST
 | |
| #include <unistd.h>
 | |
| #include <stdlib.h>
 | |
| #endif
 | |
| 
 | |
| static char Errmsg[80];
 | |
| 
 | |
| #define LOWER16BITS(X)	(X & 0177777)
 | |
| #define LOWER32BITS(X)	(X & 0xffffffff)
 | |
| 
 | |
| /***
 | |
| #define HIGHBITS(WRD, bits) ( (-1 << (64-bits)) & WRD)
 | |
| #define LOWBITS(WRD, bits) ( (-1 >> (64-bits)) & WRD)
 | |
| ****/
 | |
| 
 | |
| #define NBPBYTE		8		/* number bits per byte */
 | |
| 
 | |
| #ifndef DEBUG
 | |
| #define DEBUG	0
 | |
| #endif
 | |
| 
 | |
| /***********************************************************************
 | |
|  *
 | |
|  * 
 | |
|  * 1   2   3   4   5   6   7   8   9   10  11  12  13  14  14  15	bytes
 | |
|  * 1234567890123456789012345678901234567890123456789012345678901234	bits
 | |
|  * ________________________________________________________________	1 word
 | |
|  * <    pid       >< offset in file of this word  ><    pid       >
 | |
|  * 
 | |
|  * the words are put together where offset zero is the start.
 | |
|  * thus, offset 16 is the start of  the second full word
 | |
|  * Thus, offset 8 is in middle of word 1
 | |
|  ***********************************************************************/
 | |
| int
 | |
| datapidgen(pid, buffer, bsize, offset)
 | |
| int pid;
 | |
| char *buffer;
 | |
| int bsize;
 | |
| int offset;
 | |
| {
 | |
| #if CRAY
 | |
| 	
 | |
|    int cnt;
 | |
|    int tmp;
 | |
|    char *chr;
 | |
|    long *wptr;
 | |
|    long word;
 | |
|    int woff;	/* file offset for the word */
 | |
|    int boff;	/* buffer offset or index */
 | |
|    int num_full_words;
 | |
| 
 | |
|     num_full_words = bsize/NBPW;
 | |
|     boff = 0;
 | |
| 
 | |
|     if ( cnt=(offset % NBPW) ) {	/* partial word */
 | |
| 
 | |
| 	woff = offset - cnt;
 | |
| #if DEBUG
 | |
| printf("partial at beginning, cnt = %d, woff = %d\n", cnt, woff);
 | |
| #endif
 | |
| 
 | |
| 	word = ((LOWER16BITS(pid) << 48) | (LOWER32BITS(woff) << 16) | LOWER16BITS(pid));
 | |
| 
 | |
| 	chr = (char *)&word;
 | |
| 
 | |
| 	for (tmp=0; tmp<cnt; tmp++) {   /* skip unused bytes */
 | |
| 	    chr++;
 | |
|         }
 | |
| 
 | |
| 	for (; boff<(NBPW-cnt) && boff<bsize; boff++, chr++) {
 | |
| 	    buffer[boff] = *chr;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * full words 
 | |
|      */
 | |
| 
 | |
|     num_full_words = (bsize-boff)/NBPW;
 | |
| 	
 | |
|     woff = offset+boff;
 | |
| 	
 | |
|     for (cnt=0; cnt<num_full_words; woff += NBPW, cnt++ ) {
 | |
| 
 | |
| 	word = ((LOWER16BITS(pid) << 48) | (LOWER32BITS(woff) << 16) | LOWER16BITS(pid));
 | |
| 
 | |
| 	chr = (char *)&word;
 | |
| 	for(tmp=0; tmp<NBPW; tmp++, chr++) {
 | |
| 	    buffer[boff++] = *chr;
 | |
| 	}
 | |
| /****** Only if wptr is a word ellined
 | |
| 	wptr = (long *)&buffer[boff];
 | |
| 	*wptr = word;
 | |
| 	boff += NBPW;
 | |
| *****/
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * partial word at end of buffer
 | |
|      */
 | |
| 
 | |
|     if ( cnt=((bsize-boff) % NBPW) ) {
 | |
| #if DEBUG
 | |
| printf("partial at end\n");
 | |
| #endif
 | |
| 	word = ((LOWER16BITS(pid) << 48) | (LOWER32BITS(woff) << 16) | LOWER16BITS(pid));
 | |
| 
 | |
| 	chr = (char *)&word;
 | |
| 
 | |
| 	for (tmp=0; tmp<cnt && boff<bsize; tmp++, chr++) {
 | |
| 	    buffer[boff++] = *chr;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     return bsize;
 | |
| 
 | |
| #else
 | |
| 	return -1;	/* not support on non-64 bits word machines  */
 | |
| 
 | |
| #endif
 | |
| 
 | |
| } 
 | |
| 
 | |
| /***********************************************************************
 | |
|  *
 | |
|  *
 | |
|  ***********************************************************************/
 | |
| int
 | |
| datapidchk(pid, buffer, bsize, offset, errmsg)
 | |
| int pid;
 | |
| char *buffer;
 | |
| int bsize;
 | |
| int offset;
 | |
| char **errmsg;
 | |
| {
 | |
| #if CRAY
 | |
| 	
 | |
|    int cnt;
 | |
|    int tmp;
 | |
|    char *chr;
 | |
|    long *wptr;
 | |
|    long word;
 | |
|    int woff;	/* file offset for the word */
 | |
|    int boff;	/* buffer offset or index */
 | |
|    int num_full_words;
 | |
| 
 | |
| 
 | |
|     if ( errmsg != NULL ) {
 | |
|         *errmsg = Errmsg;
 | |
|     }
 | |
| 
 | |
| 
 | |
|     num_full_words = bsize/NBPW;
 | |
|     boff = 0;
 | |
| 
 | |
|     if ( cnt=(offset % NBPW) ) {	/* partial word */
 | |
| 	woff = offset - cnt;
 | |
| 	word = ((LOWER16BITS(pid) << 48) | (LOWER32BITS(woff) << 16) | LOWER16BITS(pid));
 | |
| 
 | |
| 	chr = (char *)&word;
 | |
| 
 | |
| 	for (tmp=0; tmp<cnt; tmp++) {   /* skip unused bytes */
 | |
| 	    chr++;
 | |
|         }
 | |
| 
 | |
| 	for (; boff<(NBPW-cnt) && boff<bsize; boff++, chr++) {
 | |
| 	    if (buffer[boff] != *chr) {
 | |
| 		sprintf(Errmsg, "Data mismatch at offset %d, exp:%#o, act:%#o",
 | |
| 		    offset+boff, *chr, buffer[boff]);
 | |
| 		return offset+boff;
 | |
| 	    }
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * full words 
 | |
|      */
 | |
| 
 | |
|     num_full_words = (bsize-boff)/NBPW;
 | |
| 	
 | |
|     woff = offset+boff;
 | |
| 	
 | |
|     for (cnt=0; cnt<num_full_words; woff += NBPW, cnt++ ) {
 | |
| 	word = ((LOWER16BITS(pid) << 48) | (LOWER32BITS(woff) << 16) | LOWER16BITS(pid));
 | |
| 
 | |
| 	chr = (char *)&word;
 | |
| 	for(tmp=0; tmp<NBPW; tmp++, boff++, chr++) {
 | |
| 	    if ( buffer[boff] != *chr ) {
 | |
| 	        sprintf(Errmsg, "Data mismatch at offset %d, exp:%#o, act:%#o",
 | |
| 	            woff, *chr, buffer[boff]);
 | |
| 	        return woff;
 | |
| 	    }
 | |
| 	}
 | |
| 
 | |
| /****** only if a word elined
 | |
| 	wptr = (long *)&buffer[boff];
 | |
| 	if ( *wptr != word ) {
 | |
| 	    sprintf(Errmsg, "Data mismatch at offset %d, exp:%#o, act:%#o",
 | |
| 	        woff, word, *wptr);
 | |
| 	    return woff;
 | |
| 	}
 | |
| 	boff += NBPW;
 | |
| ******/
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * partial word at end of buffer
 | |
|      */
 | |
| 
 | |
|     if ( cnt=((bsize-boff) % NBPW) ) {
 | |
| #if DEBUG
 | |
| printf("partial at end\n");
 | |
| #endif
 | |
| 	word = ((LOWER16BITS(pid) << 48) | (LOWER32BITS(woff) << 16) | LOWER16BITS(pid));
 | |
| 
 | |
| 	chr = (char *)&word;
 | |
| 
 | |
| 
 | |
| 	for (tmp=0; tmp<cnt && boff<bsize; boff++, tmp++, chr++) {
 | |
| 	    if ( buffer[boff] != *chr ) {
 | |
| 		sprintf(Errmsg, "Data mismatch at offset %d, exp:%#o, act:%#o",
 | |
| 		    offset+boff, *chr, buffer[boff]);
 | |
| 		return offset+boff;
 | |
| 	    }
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     sprintf(Errmsg, "all %d bytes match desired pattern", bsize);
 | |
|     return -1;      /* buffer is ok */
 | |
| 
 | |
| #else
 | |
| 	
 | |
|     if ( errmsg != NULL ) {
 | |
|         *errmsg = Errmsg;
 | |
|     }
 | |
|     sprintf(Errmsg, "Not supported on this OS.");
 | |
|     return 0;
 | |
| 
 | |
| #endif
 | |
| 
 | |
| 
 | |
| }       /* end of datapidchk */
 | |
| 
 | |
| #if UNIT_TEST
 | |
| 
 | |
| /***********************************************************************
 | |
|  * main for doing unit testing
 | |
|  ***********************************************************************/
 | |
| int
 | |
| main(ac, ag)
 | |
| int ac;
 | |
| char **ag;
 | |
| {
 | |
| 
 | |
| int size=1234;
 | |
| char *buffer;
 | |
| int ret;
 | |
| char *errmsg;
 | |
| 
 | |
|     if ((buffer=(char *)malloc(size)) == NULL ) {
 | |
|         perror("malloc");
 | |
|         exit(2);
 | |
|     }
 | |
| 
 | |
| 
 | |
|     datapidgen(-1, buffer, size, 3);
 | |
| 
 | |
| /***
 | |
| fwrite(buffer, size, 1, stdout);
 | |
| fwrite("\n", 1, 1, stdout);
 | |
| ****/
 | |
| 
 | |
|     printf("datapidgen(-1, buffer, size, 3)\n");
 | |
| 
 | |
|     ret=datapidchk(-1, buffer, size, 3, &errmsg);
 | |
|     printf("datapidchk(-1, buffer, %d, 3, &errmsg) returned %d %s\n",
 | |
|         size, ret, errmsg);
 | |
|     ret=datapidchk(-1, &buffer[1], size-1, 4, &errmsg);
 | |
|     printf("datapidchk(-1, &buffer[1], %d, 4, &errmsg) returned %d %s\n",
 | |
|         size-1, ret, errmsg);
 | |
| 
 | |
|     buffer[25]= 0x0;
 | |
|     buffer[26]= 0x0;
 | |
|     buffer[27]= 0x0;
 | |
|     buffer[28]= 0x0;
 | |
|     printf("changing char 25-28\n");
 | |
| 
 | |
|     ret=datapidchk(-1, &buffer[1], size-1, 4, &errmsg);
 | |
|     printf("datapidchk(-1, &buffer[1], %d, 4, &errmsg) returned %d %s\n",
 | |
|         size-1, ret, errmsg);
 | |
| 
 | |
| printf("------------------------------------------\n");
 | |
| 
 | |
|     datapidgen(getpid(), buffer, size, 5);
 | |
| 
 | |
| /*******
 | |
| fwrite(buffer, size, 1, stdout);
 | |
| fwrite("\n", 1, 1, stdout);
 | |
| ******/
 | |
| 
 | |
|     printf("\ndatapidgen(getpid(), buffer, size, 5)\n");
 | |
| 
 | |
|     ret=datapidchk(getpid(), buffer, size, 5, &errmsg);
 | |
|     printf("datapidchk(getpid(), buffer, %d, 5, &errmsg) returned %d %s\n",
 | |
|         size, ret, errmsg);
 | |
| 
 | |
|     ret=datapidchk(getpid(), &buffer[1], size-1, 6, &errmsg);
 | |
|     printf("datapidchk(getpid(), &buffer[1], %d, 6, &errmsg) returned %d %s\n",
 | |
|         size-1, ret, errmsg);
 | |
| 
 | |
|     buffer[25]= 0x0;
 | |
|     printf("changing char 25\n");
 | |
| 
 | |
|     ret=datapidchk(getpid(), &buffer[1], size-1, 6, &errmsg);
 | |
|     printf("datapidchk(getpid(), &buffer[1], %d, 6, &errmsg) returned %d %s\n",
 | |
|         size-1, ret, errmsg);
 | |
| 
 | |
|     exit(0);
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 |