1 /* Library function for scanning an archive file.
2 Copyright (C) 1987, 1989, 1991, 1992, 1993 Free Software Foundation, Inc.
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2, or (at your option)
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
28 /* On the sun386i and in System V rel 3, ar.h defines two different archive
29 formats depending upon whether you have defined PORTAR (normal) or PORT5AR
30 (System V Release 1). There is no default, one or the other must be defined
31 to have a nonzero value. */
33 #if (!defined (PORTAR) || PORTAR == 0) && (!defined (PORT5AR) || PORT5AR == 0)
40 /* Cray's <ar.h> apparently defines this. */
42 #define AR_HDR_SIZE (sizeof (struct ar_hdr))
45 /* SCO Unix's compiler defines both of these. */
50 /* Takes three arguments ARCHIVE, FUNCTION and ARG.
52 Open the archive named ARCHIVE, find its members one by one,
53 and for each one call FUNCTION with the following arguments:
54 archive file descriptor for reading the data,
56 member name might be truncated flag,
57 member header position in file,
58 member data position in file,
63 member protection mode,
66 The descriptor is poised to read the data of the member
67 when FUNCTION is called. It does not matter how much
70 If FUNCTION returns nonzero, we immediately return
71 what FUNCTION returned.
73 Returns -1 if archive does not exist,
74 Returns -2 if archive has invalid format.
75 Returns 0 if have scanned successfully. */
78 ar_scan (archive
, function
, arg
)
80 long int (*function
) ();
89 register int desc
= open (archive
, O_RDONLY
, 0);
95 register int nread
= read (desc
, buf
, SARMAG
);
96 if (nread
!= SARMAG
|| bcmp (buf
, ARMAG
, SARMAG
))
105 register int nread
= read (desc
, (char *) &fl_header
, FL_HSZ
);
106 if (nread
!= FL_HSZ
|| bcmp (fl_header
.fl_magic
, AIAMAG
, SAIAMAG
))
117 unsigned short int buf
;
119 register int nread
= read(desc
, &buf
, sizeof (buf
));
120 if (nread
!= sizeof (buf
) || buf
!= ARMAG
)
129 /* Now find the members one by one. */
132 register long int member_offset
= SARMAG
;
135 long int member_offset
;
136 long int last_member_offset
;
138 sscanf (fl_header
.fl_fstmoff
, "%12ld", &member_offset
);
139 sscanf (fl_header
.fl_lstmoff
, "%12ld", &last_member_offset
);
142 register long int member_offset
= sizeof (int);
144 register long int member_offset
= sizeof (unsigned short int);
145 #endif /* Not Xenix. */
152 struct ar_hdr member_header
;
158 long int data_offset
;
160 char namebuf
[sizeof member_header
.ar_name
+ 1];
162 int is_namemap
; /* Nonzero if this entry maps long names. */
168 if (lseek (desc
, member_offset
, 0) < 0)
175 #define AR_MEMHDR (AR_HDR_SIZE - sizeof (member_header._ar_name))
176 nread
= read (desc
, (char *) &member_header
, AR_MEMHDR
);
178 if (nread
!= AR_MEMHDR
)
184 sscanf (member_header
.ar_namlen
, "%4d", &name_len
);
185 nread
= read (desc
, name
, name_len
);
187 if (nread
!= name_len
)
195 sscanf (member_header
.ar_date
, "%12ld", &dateval
);
196 sscanf (member_header
.ar_uid
, "%12d", &uidval
);
197 sscanf (member_header
.ar_gid
, "%12d", &gidval
);
198 sscanf (member_header
.ar_mode
, "%12o", &eltmode
);
199 sscanf (member_header
.ar_size
, "%12ld", &eltsize
);
201 if ((data_offset
= member_offset
+ AR_MEMHDR
+ name_len
+ 2) % 2)
205 (*function
) (desc
, name
, 0,
206 member_offset
, data_offset
, eltsize
,
207 dateval
, uidval
, gidval
,
210 #else /* Not AIAMAG. */
211 nread
= read (desc
, (char *) &member_header
, AR_HDR_SIZE
);
213 /* No data left means end of file; that is OK. */
216 if (nread
!= AR_HDR_SIZE
218 || bcmp (member_header
.ar_fmag
, ARFMAG
, 2)
227 bcopy (member_header
.ar_name
, name
, sizeof member_header
.ar_name
);
229 register char *p
= name
+ sizeof member_header
.ar_name
;
232 while (p
> name
&& *--p
== ' ');
235 /* If the member name is "//" or "ARFILENAMES/" this may be
236 a list of file name mappings. The maximum file name
237 length supported by the standard archive format is 14
238 characters. This member will actually always be the
239 first or second entry in the archive, but we don't check
241 is_namemap
= (!strcmp (name
, "//")
242 || !strcmp (name
, "ARFILENAMES/"));
243 #endif /* Not AIAMAG. */
244 /* On some systems, there is a slash after each member name. */
249 /* If the member name starts with a space or a slash, this
250 is an index into the file name mappings (used by GNU ar).
251 Otherwise if the member name looks like #1/NUMBER the
252 real member name appears in the element data (used by
255 && (name
[0] == ' ' || name
[0] == '/')
258 name
= namemap
+ atoi (name
+ 1);
261 else if (name
[0] == '#'
265 int namesize
= atoi (name
+ 3);
267 name
= (char *) alloca (namesize
+ 1);
268 nread
= read (desc
, name
, namesize
);
269 if (nread
!= namesize
)
274 name
[namesize
] = '\0';
278 #endif /* Not AIAMAG. */
282 sscanf (member_header
.ar_mode
, "%o", &eltmode
);
283 eltsize
= atol (member_header
.ar_size
);
285 eltmode
= (unsigned short int) member_header
.ar_mode
;
286 eltsize
= member_header
.ar_size
;
287 #endif /* Not Xenix. */
290 (*function
) (desc
, name
, ! long_name
, member_offset
,
291 member_offset
+ AR_HDR_SIZE
, eltsize
,
293 atol (member_header
.ar_date
),
294 atoi (member_header
.ar_uid
),
295 atoi (member_header
.ar_gid
),
297 member_header
.ar_date
,
298 member_header
.ar_uid
,
299 member_header
.ar_gid
,
300 #endif /* Not Xenix. */
312 if (member_offset
== last_member_offset
)
313 /* End of the chain. */
316 sscanf (member_header
.ar_nxtmem
, "%12ld", &member_offset
);
318 if (lseek (desc
, member_offset
, 0) != member_offset
)
325 /* If this member maps archive names, we must read it in. The
326 name map will always precede any members whose names must
333 namemap
= (char *) alloca (eltsize
);
334 nread
= read (desc
, namemap
, eltsize
);
335 if (nread
!= eltsize
)
341 /* The names are separated by newlines. Some formats have
342 a trailing slash. Null terminate the strings for
344 limit
= namemap
+ eltsize
;
345 for (clear
= namemap
; clear
< limit
; clear
++)
350 if (clear
[-1] == '/')
358 member_offset
+= AR_HDR_SIZE
+ eltsize
;
359 if (member_offset
% 2 != 0)
369 /* Return nonzero iff NAME matches MEM.
370 If TRUNCATED is nonzero, MEM may be truncated to
371 sizeof (struct ar_hdr.ar_name) - 1. */
374 ar_name_equal (name
, mem
, truncated
)
380 p
= rindex (name
, '/');
384 /* We no longer use this kludge, since we
385 now support long archive member names. */
387 #if 0 && !defined (AIAMAG) && !defined (APOLLO)
390 /* `reallylongname.o' matches `reallylongnam.o'.
391 If member names have a trailing slash, that's `reallylongna.o'. */
394 unsigned int max
= sizeof (h
.ar_name
);
395 unsigned int namelen
, memlen
;
397 if (strncmp (name
, mem
, max
- 3))
400 namelen
= strlen (name
);
401 memlen
= strlen (mem
);
403 if (namelen
> memlen
&& memlen
>= max
- 1
404 && name
[namelen
- 2] == '.' && name
[namelen
- 1] == 'o'
405 && mem
[memlen
- 2] == '.' && mem
[memlen
- 1] == 'o')
408 if (namelen
!= memlen
)
411 return (namelen
< max
- 3 || !strcmp (name
+ max
- 3, mem
+ max
- 3));
414 #else /* AIX or APOLLO. */
419 /* TRUNCATED should never be set on this system. */
423 return !strncmp (name
, mem
,
424 sizeof (hdr
.ar_name
) - 1);
428 return !strcmp (name
, mem
);
435 ar_member_pos (desc
, mem
, truncated
,
436 hdrpos
, datapos
, size
, date
, uid
, gid
, mode
, name
)
440 long int hdrpos
, datapos
, size
, date
;
444 if (!ar_name_equal (name
, mem
, truncated
))
449 /* Set date of member MEMNAME in archive ARNAME to current time.
450 Returns 0 if successful,
451 -1 if file ARNAME does not exist,
452 -2 if not a valid archive,
453 -3 if other random system call error (including file read-only),
454 1 if valid but member MEMNAME does not exist. */
457 ar_member_touch (arname
, memname
)
458 char *arname
, *memname
;
460 register long int pos
= ar_scan (arname
, ar_member_pos
, (long int) memname
);
462 struct ar_hdr ar_hdr
;
471 fd
= open (arname
, O_RDWR
, 0666);
474 /* Read in this member's header */
475 if (lseek (fd
, pos
, 0) < 0)
477 if (AR_HDR_SIZE
!= read (fd
, (char *) &ar_hdr
, AR_HDR_SIZE
))
479 /* Write back the header, thus touching the archive file. */
480 if (lseek (fd
, pos
, 0) < 0)
482 if (AR_HDR_SIZE
!= write (fd
, (char *) &ar_hdr
, AR_HDR_SIZE
))
484 /* The file's mtime is the time we we want. */
485 fstat (fd
, &statbuf
);
486 #if defined(ARFMAG) || defined(AIAMAG)
487 /* Advance member's time to that time */
488 for (i
= 0; i
< sizeof ar_hdr
.ar_date
; i
++)
489 ar_hdr
.ar_date
[i
] = ' ';
490 sprintf (ar_hdr
.ar_date
, "%ld", (long int) statbuf
.st_mtime
);
492 ar_hdr
.ar_date
[strlen(ar_hdr
.ar_date
)] = ' ';
495 ar_hdr
.ar_date
= statbuf
.st_mtime
;
497 /* Write back this member's header */
498 if (lseek (fd
, pos
, 0) < 0)
500 if (AR_HDR_SIZE
!= write (fd
, (char *) &ar_hdr
, AR_HDR_SIZE
))
515 describe_member (desc
, name
, truncated
,
516 hdrpos
, datapos
, size
, date
, uid
, gid
, mode
)
520 long int hdrpos
, datapos
, size
, date
;
523 extern char *ctime ();
525 printf ("Member `%s'%s: %ld bytes at %ld (%ld).\n",
526 name
, truncated
? " (name might be truncated)" : "",
527 size
, hdrpos
, datapos
);
528 printf (" Date %s", ctime (&date
));
529 printf (" uid = %d, gid = %d, mode = 0%o.\n", uid
, gid
, mode
);
538 ar_scan (argv
[1], describe_member
);
544 #endif /* NO_ARCHIVES. */