1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270
|
/*
* $Header: /cvsroot/arc/arc/arcio.c,v 1.4 2005/10/12 15:22:18 highlandsun Exp $
*/
/* ARC - Archive utility - ARCIO
Version 2.50, created on 04/22/87 at 13:25:20
(C) COPYRIGHT 1985,86 by System Enhancement Associates; ALL RIGHTS RESERVED
By: Thom Henderson
Description:
This file contains the file I/O routines used to manipulate
an archive.
Language:
Computer Innovations Optimizing C86
*/
#include <stdio.h>
#include "arc.h"
#if _MTS
#include <mts.h>
#endif
#include <string.h>
VOID arcdie();
int
readhdr(hdr, f) /* read a header from an archive */
struct heads *hdr; /* storage for header */
FILE *f; /* archive to read header from */
{
#if !MSDOS
unsigned char dummy[28];
int i;
#endif
char name[FNLEN]; /* filename buffer */
int try = 0;/* retry counter */
static int first = 1; /* true only on first read */
if (!f) /* if archive didn't open */
return 0; /* then pretend it's the end */
hdrver = fgetc(f);
if (feof(f)) /* if no more data */
return 0; /* then signal end of archive */
if (hdrver != ARCMARK) { /* check archive validity */
if (warn) {
printf("An entry in %s has a bad header.\n", arcname);
nerrs++;
}
printf("hdrver: %x\n", hdrver);
while (!feof(f)) {
try++;
if (fgetc(f) == ARCMARK) {
int dummy;
/* ungetc is guaranteed to succeed for a single char,
* no need to check result
*/
dummy = ungetc(hdrver = fgetc(f), f);
if (!(hdrver & 0x80) && hdrver <= ARCVER)
break;
}
}
if (feof(f) && first)
arcdie("%s is not an archive", arcname);
if (changing && warn)
arcdie("%s is corrupted -- changes disallowed", arcname);
if (warn)
printf(" %d bytes skipped.\n", try);
if (feof(f))
return 0;
}
hdrver = fgetc(f); /* get header version */
if (hdrver & 0x80) /* sign bit? negative? */
arcdie("Invalid header in archive %s", arcname);
if (hdrver == 0)
return 0; /* note our end of archive marker */
if (hdrver > ARCVER) {
if (fread(name, sizeof(char), FNLEN, f) != FNLEN)
arcdie("%s was truncated", arcname);
#if _MTS
atoe(name, strlen(name));
#endif
printf("I don't know how to handle file %s in archive %s\n",
name, arcname);
printf("I think you need a newer version of ARC.\n");
exit(1);
}
/* amount to read depends on header type */
if (hdrver == 1) { /* old style is shorter */
if (fread(hdr, sizeof(struct heads) - sizeof(long int), 1, f) != 1)
arcdie("%s was truncated", arcname);
hdrver = 2; /* convert header to new format */
hdr->length = hdr->size; /* size is same when not
* packed */
} else
#if MSDOS
if (fread(hdr, sizeof(struct heads), 1, f) != 1)
arcdie("%s was truncated", arcname);
#else
if (fread(dummy, 27, 1, f) != 1)
arcdie("%s was truncated", arcname);
for (i = 0; i < FNLEN; hdr->name[i] = dummy[i], i++);
#if _MTS
(void) atoe(hdr->name, strlen(hdr->name));
#endif
for (i = 0, hdr->size=0; i<4; hdr->size<<=8, hdr->size += dummy[16-i], i++);
hdr->date = (short) ((dummy[18] << 8) + dummy[17]);
hdr->time = (short) ((dummy[20] << 8) + dummy[19]);
hdr->crc = (short) ((dummy[22] << 8) + dummy[21]);
for (i = 0, hdr->length=0; i<4; hdr->length<<=8, hdr->length += dummy[26-i], i++);
#endif
if (hdr->date > olddate
|| (hdr->date == olddate && hdr->time > oldtime)) {
olddate = hdr->date;
oldtime = hdr->time;
}
first = 0;
return 1; /* we read something */
}
VOID
put_int(number, f) /* write a 2 byte integer */
short number;
FILE *f;
{
fputc((char) (number & 255), f);
fputc((char) (number >> 8), f);
}
VOID
put_long(number, f) /* write a 4 byte integer */
long number;
FILE *f;
{
put_int((short) (number & 0xFFFF), f);
put_int((short) (number >> 16), f);
}
VOID
writehdr(hdr, f) /* write a header to an archive */
struct heads *hdr; /* header to write */
FILE *f; /* archive to write to */
{
fputc(ARCMARK, f); /* write out the mark of ARC */
fputc(hdrver, f); /* write out the header version */
if (!hdrver) /* if that's the end */
return; /* then write no more */
#if MSDOS
if (fwrite(hdr, sizeof(struct heads), 1, f) != 1)
arcdie("%s out of space", arcname);
#else
/* byte/word ordering hassles... */
#if _MTS
etoa(hdr->name, strlen(hdr->name));
#endif
if (fwrite(hdr->name, 1, FNLEN, f) != FNLEN)
arcdie("%s out of space", arcname);
put_long(hdr->size, f);
put_int(hdr->date, f);
put_int(hdr->time, f);
put_int(hdr->crc, f);
put_long(hdr->length, f);
#endif
/* note the newest file for updating the archive timestamp */
if (hdr->date > arcdate
|| (hdr->date == arcdate && hdr->time > arctime)) {
arcdate = hdr->date;
arctime = hdr->time;
}
}
extern char *pinbuf; /* general purpose input buffer */
/*
* NOTE: The filecopy() function is used to move large numbers of bytes from
* one file to another. This particular version has been modified to improve
* performance in Computer Innovations C86 version 2.3 in the small memory
* model. It may not work as expected with other compilers or libraries, or
* indeed with different versions of the CI-C86 compiler and library, or with
* the same version in a different memory model.
*
* The following is a functional equivalent to the filecopy() routine that
* should work properly on any system using any compiler, albeit at the cost
* of reduced performance:
*
* filecopy(f,t,size)
* FILE *f, *t; long size;
* {
* while(size--)
* putc_tst(fgetc(f),t);
* }
*/
#if MSDOS
#include <fileio2.h>
filecopy(f, t, size) /* bulk file copier */
FILE *f, *t; /* files from and to */
long size; /* bytes to copy */
{
unsigned int bufl; /* buffer length */
unsigned int coreleft(); /* space available reporter */
unsigned int cpy; /* bytes being copied */
long floc, tloc, fseek(); /* file pointers, setter */
struct regval reg; /* registers for DOS calls */
bufl = (MYBUF > size) ? (u_int) size : MYBUF;
floc = fseek(f, 0L, 1); /* reset I/O system */
tloc = fseek(t, 0L, 1);
segread(®.si); /* set segment registers for DOS */
while (size > 0) { /* while more to copy */
reg.ax = 0x3F00;/* read from handle */
reg.bx = filehand(f);
reg.cx = bufl < size ? bufl : size; /* amount to read */
reg.dx = pinbuf;
if (sysint21(®, ®) & 1)
arcdie("Read fail");
cpy = reg.ax; /* amount actually read */
reg.ax = 0x4000;/* write to handle */
reg.bx = filehand(t);
reg.cx = cpy;
reg.dx = pinbuf;
sysint21(®, ®);
if (reg.ax != cpy)
arcdie("Write fail (disk full?)");
size -= (long) cpy;
}
}
#else
VOID
filecopy(f, t, size) /* bulk file copier */
FILE *f, *t; /* files from and to */
long size; /* bytes to copy */
{
unsigned int bufl; /* buffer length */
unsigned int cpy; /* bytes being copied */
bufl = (MYBUF > size) ? (u_int) size : MYBUF;
while (size > 0) {
cpy = fread(pinbuf, sizeof(char), bufl, f);
if (fwrite(pinbuf, sizeof(char), cpy, t) != cpy)
arcdie("Write fail (no space?)");
size -= cpy;
if (bufl > size)
bufl = size;
if (ferror(f))
arcdie("Unexpected EOF copying archive");
if (!cpy) break;
}
}
#endif
|