General improvements to Rockridge parsing and ISO9660 format detection.
This commit is contained in:
parent
6a6f3afd42
commit
10711278cc
@ -138,6 +138,15 @@ __FBSDID("$FreeBSD$");
|
|||||||
#define PVD_reserved4_size 1
|
#define PVD_reserved4_size 1
|
||||||
#define PVD_application_data_offset (PVD_reserved4_offset + PVD_reserved4_size)
|
#define PVD_application_data_offset (PVD_reserved4_offset + PVD_reserved4_size)
|
||||||
#define PVD_application_data_size 512
|
#define PVD_application_data_size 512
|
||||||
|
#define PVD_reserved5_offset (PVD_application_data_offset + PVD_application_data_size)
|
||||||
|
#define PVD_reserved5_size (2048 - PVD_reserved5_offset)
|
||||||
|
|
||||||
|
/* TODO: It would make future maintenance easier to just hardcode the
|
||||||
|
* above values. In particular, ECMA119 states the offsets as part of
|
||||||
|
* the standard. That would eliminate the need for the following check.*/
|
||||||
|
#if PVD_reserved5_offset != 1395
|
||||||
|
#error PVD offset and size definitions are wrong.
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Structure of an on-disk directory record. */
|
/* Structure of an on-disk directory record. */
|
||||||
/* Note: ISO9660 stores each multi-byte integer twice, once in
|
/* Note: ISO9660 stores each multi-byte integer twice, once in
|
||||||
@ -178,17 +187,20 @@ struct file_info {
|
|||||||
uint64_t size; /* File size in bytes. */
|
uint64_t size; /* File size in bytes. */
|
||||||
uint64_t ce_offset; /* Offset of CE */
|
uint64_t ce_offset; /* Offset of CE */
|
||||||
uint64_t ce_size; /* Size of CE */
|
uint64_t ce_size; /* Size of CE */
|
||||||
|
time_t birthtime; /* File created time. */
|
||||||
time_t mtime; /* File last modified time. */
|
time_t mtime; /* File last modified time. */
|
||||||
time_t atime; /* File last accessed time. */
|
time_t atime; /* File last accessed time. */
|
||||||
time_t ctime; /* File creation time. */
|
time_t ctime; /* File attribute change time. */
|
||||||
uint64_t rdev; /* Device number */
|
uint64_t rdev; /* Device number */
|
||||||
mode_t mode;
|
mode_t mode;
|
||||||
uid_t uid;
|
uid_t uid;
|
||||||
gid_t gid;
|
gid_t gid;
|
||||||
ino_t inode;
|
ino_t inode;
|
||||||
int nlinks;
|
int nlinks;
|
||||||
char *name; /* Null-terminated filename. */
|
struct archive_string name; /* Pathname */
|
||||||
|
char name_continues; /* Non-zero if name continues */
|
||||||
struct archive_string symlink;
|
struct archive_string symlink;
|
||||||
|
char symlink_continues; /* Non-zero if link continues */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -210,6 +222,7 @@ struct iso9660 {
|
|||||||
|
|
||||||
uint64_t current_position;
|
uint64_t current_position;
|
||||||
ssize_t logical_block_size;
|
ssize_t logical_block_size;
|
||||||
|
uint64_t volume_size; /* Total size of volume in bytes. */
|
||||||
|
|
||||||
off_t entry_sparse_offset;
|
off_t entry_sparse_offset;
|
||||||
int64_t entry_bytes_remaining;
|
int64_t entry_bytes_remaining;
|
||||||
@ -224,7 +237,7 @@ static int archive_read_format_iso9660_read_data_skip(struct archive_read *);
|
|||||||
static int archive_read_format_iso9660_read_header(struct archive_read *,
|
static int archive_read_format_iso9660_read_header(struct archive_read *,
|
||||||
struct archive_entry *);
|
struct archive_entry *);
|
||||||
static const char *build_pathname(struct archive_string *, struct file_info *);
|
static const char *build_pathname(struct archive_string *, struct file_info *);
|
||||||
#ifdef DEBUG
|
#if DEBUG
|
||||||
static void dump_isodirrec(FILE *, const unsigned char *isodirrec);
|
static void dump_isodirrec(FILE *, const unsigned char *isodirrec);
|
||||||
#endif
|
#endif
|
||||||
static time_t time_from_tm(struct tm *);
|
static time_t time_from_tm(struct tm *);
|
||||||
@ -240,6 +253,12 @@ static struct file_info *
|
|||||||
static void parse_rockridge(struct iso9660 *iso9660,
|
static void parse_rockridge(struct iso9660 *iso9660,
|
||||||
struct file_info *file, const unsigned char *start,
|
struct file_info *file, const unsigned char *start,
|
||||||
const unsigned char *end);
|
const unsigned char *end);
|
||||||
|
static void parse_rockridge_NM1(struct file_info *,
|
||||||
|
const unsigned char *, int);
|
||||||
|
static void parse_rockridge_SL1(struct file_info *,
|
||||||
|
const unsigned char *, int);
|
||||||
|
static void parse_rockridge_TF1(struct file_info *,
|
||||||
|
const unsigned char *, int);
|
||||||
static void release_file(struct iso9660 *, struct file_info *);
|
static void release_file(struct iso9660 *, struct file_info *);
|
||||||
static unsigned toi(const void *p, int n);
|
static unsigned toi(const void *p, int n);
|
||||||
|
|
||||||
@ -316,13 +335,61 @@ static int
|
|||||||
isPVD(struct iso9660 *iso9660, const unsigned char *h)
|
isPVD(struct iso9660 *iso9660, const unsigned char *h)
|
||||||
{
|
{
|
||||||
struct file_info *file;
|
struct file_info *file;
|
||||||
|
int i;
|
||||||
|
|
||||||
if (h[0] != 1)
|
/* Type of the Primary Volume Descriptor must be 1. */
|
||||||
return (0);
|
if (h[PVD_type_offset] != 1)
|
||||||
if (memcmp(h+1, "CD001", 5) != 0)
|
|
||||||
return (0);
|
return (0);
|
||||||
|
|
||||||
|
/* ID must be "CD001" */
|
||||||
|
if (memcmp(h + PVD_id_offset, "CD001", 5) != 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
/* PVD version must be 1. */
|
||||||
|
if (h[PVD_version_offset] != 1)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
/* Reserved field must be 0. */
|
||||||
|
if (h[PVD_reserved1_offset] != 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
/* Reserved field must be 0. */
|
||||||
|
for (i = 0; i < PVD_reserved2_size; ++i)
|
||||||
|
if (h[PVD_reserved2_offset + i] != 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
/* Reserved field must be 0. */
|
||||||
|
for (i = 0; i < PVD_reserved3_size; ++i)
|
||||||
|
if (h[PVD_reserved3_offset + i] != 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
/* Logical block size must be > 0. */
|
||||||
|
/* I've looked at Ecma 119 and can't find any stronger
|
||||||
|
* restriction on this field. */
|
||||||
iso9660->logical_block_size = toi(h + PVD_logical_block_size_offset, 2);
|
iso9660->logical_block_size = toi(h + PVD_logical_block_size_offset, 2);
|
||||||
|
if (iso9660->logical_block_size <= 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
iso9660->volume_size = iso9660->logical_block_size
|
||||||
|
* (uint64_t)toi(h + PVD_volume_space_size_offset, 4);
|
||||||
|
|
||||||
|
/* File structure version must be 1 for ISO9660/ECMA119. */
|
||||||
|
if (h[PVD_file_structure_version_offset] != 1)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
|
||||||
|
/* Reserved field must be 0. */
|
||||||
|
for (i = 0; i < PVD_reserved4_size; ++i)
|
||||||
|
if (h[PVD_reserved4_offset + i] != 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
/* Reserved field must be 0. */
|
||||||
|
for (i = 0; i < PVD_reserved5_size; ++i)
|
||||||
|
if (h[PVD_reserved5_offset + i] != 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
/* XXX TODO: Check other values for sanity; reject more
|
||||||
|
* malformed PVDs. XXX */
|
||||||
|
|
||||||
/* Store the root directory in the pending list. */
|
/* Store the root directory in the pending list. */
|
||||||
file = parse_file_info(iso9660, NULL, h + PVD_root_directory_record_offset);
|
file = parse_file_info(iso9660, NULL, h + PVD_root_directory_record_offset);
|
||||||
@ -353,12 +420,22 @@ archive_read_format_iso9660_read_header(struct archive_read *a,
|
|||||||
iso9660->entry_bytes_remaining = file->size;
|
iso9660->entry_bytes_remaining = file->size;
|
||||||
iso9660->entry_sparse_offset = 0; /* Offset for sparse-file-aware clients. */
|
iso9660->entry_sparse_offset = 0; /* Offset for sparse-file-aware clients. */
|
||||||
|
|
||||||
|
if (file->offset + file->size > iso9660->volume_size) {
|
||||||
|
archive_set_error(&a->archive, ARCHIVE_ERRNO_MISC,
|
||||||
|
"File is beyond end-of-media: %s", file->name);
|
||||||
|
iso9660->entry_bytes_remaining = 0;
|
||||||
|
iso9660->entry_sparse_offset = 0;
|
||||||
|
release_file(iso9660, file);
|
||||||
|
return (ARCHIVE_WARN);
|
||||||
|
}
|
||||||
|
|
||||||
/* Set up the entry structure with information about this entry. */
|
/* Set up the entry structure with information about this entry. */
|
||||||
archive_entry_set_mode(entry, file->mode);
|
archive_entry_set_mode(entry, file->mode);
|
||||||
archive_entry_set_uid(entry, file->uid);
|
archive_entry_set_uid(entry, file->uid);
|
||||||
archive_entry_set_gid(entry, file->gid);
|
archive_entry_set_gid(entry, file->gid);
|
||||||
archive_entry_set_nlink(entry, file->nlinks);
|
archive_entry_set_nlink(entry, file->nlinks);
|
||||||
archive_entry_set_ino(entry, file->inode);
|
archive_entry_set_ino(entry, file->inode);
|
||||||
|
archive_entry_set_birthtime(entry, file->birthtime, 0);
|
||||||
archive_entry_set_mtime(entry, file->mtime, 0);
|
archive_entry_set_mtime(entry, file->mtime, 0);
|
||||||
archive_entry_set_ctime(entry, file->ctime, 0);
|
archive_entry_set_ctime(entry, file->ctime, 0);
|
||||||
archive_entry_set_atime(entry, file->atime, 0);
|
archive_entry_set_atime(entry, file->atime, 0);
|
||||||
@ -534,13 +611,7 @@ parse_file_info(struct iso9660 *iso9660, struct file_info *parent,
|
|||||||
file->mtime = isodate7(isodirrec + DR_date_offset);
|
file->mtime = isodate7(isodirrec + DR_date_offset);
|
||||||
file->ctime = file->atime = file->mtime;
|
file->ctime = file->atime = file->mtime;
|
||||||
name_len = (size_t)*(const unsigned char *)(isodirrec + DR_name_len_offset);
|
name_len = (size_t)*(const unsigned char *)(isodirrec + DR_name_len_offset);
|
||||||
file->name = (char *)malloc(name_len + 1);
|
archive_strncpy(&file->name, isodirrec + DR_name_offset, name_len);
|
||||||
if (file->name == NULL) {
|
|
||||||
free(file);
|
|
||||||
return (NULL);
|
|
||||||
}
|
|
||||||
memcpy(file->name, isodirrec + DR_name_offset, name_len);
|
|
||||||
file->name[name_len] = '\0';
|
|
||||||
flags = *(isodirrec + DR_flags_offset);
|
flags = *(isodirrec + DR_flags_offset);
|
||||||
if (flags & 0x02)
|
if (flags & 0x02)
|
||||||
file->mode = AE_IFDIR | 0700;
|
file->mode = AE_IFDIR | 0700;
|
||||||
@ -560,7 +631,7 @@ parse_file_info(struct iso9660 *iso9660, struct file_info *parent,
|
|||||||
parse_rockridge(iso9660, file, rr_start, rr_end);
|
parse_rockridge(iso9660, file, rr_start, rr_end);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG
|
#if DEBUG
|
||||||
/* DEBUGGING: Warn about attributes I don't yet fully support. */
|
/* DEBUGGING: Warn about attributes I don't yet fully support. */
|
||||||
if ((flags & ~0x02) != 0) {
|
if ((flags & ~0x02) != 0) {
|
||||||
fprintf(stderr, "\n ** Unrecognized flag: ");
|
fprintf(stderr, "\n ** Unrecognized flag: ");
|
||||||
@ -584,7 +655,6 @@ parse_file_info(struct iso9660 *iso9660, struct file_info *parent,
|
|||||||
fprintf(stderr, "\n");
|
fprintf(stderr, "\n");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return (file);
|
return (file);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -624,6 +694,7 @@ parse_rockridge(struct iso9660 *iso9660, struct file_info *file,
|
|||||||
while (p + 4 < end /* Enough space for another entry. */
|
while (p + 4 < end /* Enough space for another entry. */
|
||||||
&& p[0] >= 'A' && p[0] <= 'Z' /* Sanity-check 1st char of name. */
|
&& p[0] >= 'A' && p[0] <= 'Z' /* Sanity-check 1st char of name. */
|
||||||
&& p[1] >= 'A' && p[1] <= 'Z' /* Sanity-check 2nd char of name. */
|
&& p[1] >= 'A' && p[1] <= 'Z' /* Sanity-check 2nd char of name. */
|
||||||
|
&& p[2] >= 4 /* Sanity-check length. */
|
||||||
&& p + p[2] <= end) { /* Sanity-check length. */
|
&& p + p[2] <= end) { /* Sanity-check length. */
|
||||||
const unsigned char *data = p + 4;
|
const unsigned char *data = p + 4;
|
||||||
int data_length = p[2] - 4;
|
int data_length = p[2] - 4;
|
||||||
@ -636,61 +707,54 @@ parse_rockridge(struct iso9660 *iso9660, struct file_info *file,
|
|||||||
*/
|
*/
|
||||||
switch(p[0]) {
|
switch(p[0]) {
|
||||||
case 'C':
|
case 'C':
|
||||||
if (p[0] == 'C' && p[1] == 'E' && version == 1) {
|
if (p[0] == 'C' && p[1] == 'E') {
|
||||||
/*
|
if (version == 1 && data_length == 24) {
|
||||||
* CE extension comprises:
|
/*
|
||||||
* 8 byte sector containing extension
|
* CE extension comprises:
|
||||||
* 8 byte offset w/in above sector
|
* 8 byte sector containing extension
|
||||||
* 8 byte length of continuation
|
* 8 byte offset w/in above sector
|
||||||
*/
|
* 8 byte length of continuation
|
||||||
file->ce_offset = toi(data, 4)
|
*/
|
||||||
* iso9660->logical_block_size
|
file->ce_offset = (uint64_t)toi(data, 4)
|
||||||
+ toi(data + 8, 4);
|
* iso9660->logical_block_size
|
||||||
file->ce_size = toi(data + 16, 4);
|
+ toi(data + 8, 4);
|
||||||
|
file->ce_size = toi(data + 16, 4);
|
||||||
|
/* If the result is rediculous,
|
||||||
|
* ignore it. */
|
||||||
|
if (file->ce_offset + file->ce_size
|
||||||
|
> iso9660->volume_size) {
|
||||||
|
file->ce_offset = 0;
|
||||||
|
file->ce_size = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
/* FALLTHROUGH */
|
/* FALLTHROUGH */
|
||||||
case 'N':
|
case 'N':
|
||||||
if (p[0] == 'N' && p[1] == 'M' && version == 1
|
if (p[0] == 'N' && p[1] == 'M') {
|
||||||
&& *data == 0) {
|
if (version == 1)
|
||||||
/* NM extension with flag byte == 0 */
|
parse_rockridge_NM1(file,
|
||||||
/*
|
data, data_length);
|
||||||
* NM extension comprises:
|
|
||||||
* one byte flag
|
|
||||||
* rest is long name
|
|
||||||
*/
|
|
||||||
/* TODO: Obey flags. */
|
|
||||||
char *old_name = file->name;
|
|
||||||
|
|
||||||
data++; /* Skip flag byte. */
|
|
||||||
data_length--;
|
|
||||||
file->name = (char *)malloc(data_length + 1);
|
|
||||||
if (file->name != NULL) {
|
|
||||||
free(old_name);
|
|
||||||
memcpy(file->name, data, data_length);
|
|
||||||
file->name[data_length] = '\0';
|
|
||||||
} else
|
|
||||||
file->name = old_name;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
/* FALLTHROUGH */
|
/* FALLTHROUGH */
|
||||||
case 'P':
|
case 'P':
|
||||||
if (p[0] == 'P' && p[1] == 'D' && version == 1) {
|
if (p[0] == 'P' && p[1] == 'D') {
|
||||||
/*
|
/*
|
||||||
* PD extension is padding;
|
* PD extension is padding;
|
||||||
* contents are always ignored.
|
* contents are always ignored.
|
||||||
*/
|
*/
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (p[0] == 'P' && p[1] == 'N' && version == 1) {
|
if (p[0] == 'P' && p[1] == 'N') {
|
||||||
if (data_length == 16) {
|
if (version == 1 && data_length == 16) {
|
||||||
file->rdev = toi(data,4);
|
file->rdev = toi(data,4);
|
||||||
file->rdev <<= 32;
|
file->rdev <<= 32;
|
||||||
file->rdev |= toi(data + 8, 4);
|
file->rdev |= toi(data + 8, 4);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (p[0] == 'P' && p[1] == 'X' && version == 1) {
|
if (p[0] == 'P' && p[1] == 'X') {
|
||||||
/*
|
/*
|
||||||
* PX extension comprises:
|
* PX extension comprises:
|
||||||
* 8 bytes for mode,
|
* 8 bytes for mode,
|
||||||
@ -699,12 +763,22 @@ parse_rockridge(struct iso9660 *iso9660, struct file_info *file,
|
|||||||
* 8 bytes for gid,
|
* 8 bytes for gid,
|
||||||
* 8 bytes for inode.
|
* 8 bytes for inode.
|
||||||
*/
|
*/
|
||||||
if (data_length == 32) {
|
if (version == 1) {
|
||||||
file->mode = toi(data, 4);
|
if (data_length >= 8)
|
||||||
file->nlinks = toi(data + 8, 4);
|
file->mode
|
||||||
file->uid = toi(data + 16, 4);
|
= toi(data, 4);
|
||||||
file->gid = toi(data + 24, 4);
|
if (data_length >= 16)
|
||||||
file->inode = toi(data + 32, 4);
|
file->nlinks
|
||||||
|
= toi(data + 8, 4);
|
||||||
|
if (data_length >= 24)
|
||||||
|
file->uid
|
||||||
|
= toi(data + 16, 4);
|
||||||
|
if (data_length >= 32)
|
||||||
|
file->gid
|
||||||
|
= toi(data + 24, 4);
|
||||||
|
if (data_length >= 40)
|
||||||
|
file->inode
|
||||||
|
= toi(data + 32, 4);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -721,56 +795,14 @@ parse_rockridge(struct iso9660 *iso9660, struct file_info *file,
|
|||||||
}
|
}
|
||||||
/* FALLTHROUGH */
|
/* FALLTHROUGH */
|
||||||
case 'S':
|
case 'S':
|
||||||
if (p[0] == 'S' && p[1] == 'L' && version == 1
|
if (p[0] == 'S' && p[1] == 'L') {
|
||||||
&& *data == 0) {
|
if (version == 1)
|
||||||
int cont = 1;
|
parse_rockridge_SL1(file,
|
||||||
/* SL extension with flags == 0 */
|
data, data_length);
|
||||||
/* TODO: handle non-zero flag values. */
|
|
||||||
data++; /* Skip flag byte. */
|
|
||||||
data_length--;
|
|
||||||
while (data_length > 0) {
|
|
||||||
unsigned char flag = *data++;
|
|
||||||
unsigned char nlen = *data++;
|
|
||||||
data_length -= 2;
|
|
||||||
|
|
||||||
if (cont == 0)
|
|
||||||
archive_strcat(&file->symlink, "/");
|
|
||||||
cont = 0;
|
|
||||||
|
|
||||||
switch(flag) {
|
|
||||||
case 0x01: /* Continue */
|
|
||||||
archive_strncat(&file->symlink,
|
|
||||||
(const char *)data, nlen);
|
|
||||||
cont = 1;
|
|
||||||
break;
|
|
||||||
case 0x02: /* Current */
|
|
||||||
archive_strcat(&file->symlink, ".");
|
|
||||||
break;
|
|
||||||
case 0x04: /* Parent */
|
|
||||||
archive_strcat(&file->symlink, "..");
|
|
||||||
break;
|
|
||||||
case 0x08: /* Root */
|
|
||||||
case 0x10: /* Volume root */
|
|
||||||
archive_string_empty(&file->symlink);
|
|
||||||
break;
|
|
||||||
case 0x20: /* Hostname */
|
|
||||||
archive_strcat(&file->symlink, "hostname");
|
|
||||||
break;
|
|
||||||
case 0:
|
|
||||||
archive_strncat(&file->symlink,
|
|
||||||
(const char *)data, nlen);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
/* TODO: issue a warning ? */
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
data += nlen;
|
|
||||||
data_length -= nlen;
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (p[0] == 'S' && p[1] == 'P'
|
if (p[0] == 'S' && p[1] == 'P'
|
||||||
&& version == 1 && data_length == 7
|
&& version == 1 && data_length == 3
|
||||||
&& data[0] == (unsigned char)'\xbe'
|
&& data[0] == (unsigned char)'\xbe'
|
||||||
&& data[1] == (unsigned char)'\xef') {
|
&& data[1] == (unsigned char)'\xef') {
|
||||||
/*
|
/*
|
||||||
@ -806,62 +838,20 @@ parse_rockridge(struct iso9660 *iso9660, struct file_info *file,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
case 'T':
|
case 'T':
|
||||||
if (p[0] == 'T' && p[1] == 'F' && version == 1) {
|
if (p[0] == 'T' && p[1] == 'F') {
|
||||||
char flag = data[0];
|
if (version == 1)
|
||||||
/*
|
parse_rockridge_TF1(file,
|
||||||
* TF extension comprises:
|
data, data_length);
|
||||||
* one byte flag
|
|
||||||
* create time (optional)
|
|
||||||
* modify time (optional)
|
|
||||||
* access time (optional)
|
|
||||||
* attribute time (optional)
|
|
||||||
* Time format and presence of fields
|
|
||||||
* is controlled by flag bits.
|
|
||||||
*/
|
|
||||||
data++;
|
|
||||||
if (flag & 0x80) {
|
|
||||||
/* Use 17-byte time format. */
|
|
||||||
if (flag & 1) /* Create time. */
|
|
||||||
data += 17;
|
|
||||||
if (flag & 2) { /* Modify time. */
|
|
||||||
file->mtime = isodate17(data);
|
|
||||||
data += 17;
|
|
||||||
}
|
|
||||||
if (flag & 4) { /* Access time. */
|
|
||||||
file->atime = isodate17(data);
|
|
||||||
data += 17;
|
|
||||||
}
|
|
||||||
if (flag & 8) { /* Attribute time. */
|
|
||||||
file->ctime = isodate17(data);
|
|
||||||
data += 17;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
/* Use 7-byte time format. */
|
|
||||||
if (flag & 1) /* Create time. */
|
|
||||||
data += 7;
|
|
||||||
if (flag & 2) { /* Modify time. */
|
|
||||||
file->mtime = isodate7(data);
|
|
||||||
data += 7;
|
|
||||||
}
|
|
||||||
if (flag & 4) { /* Access time. */
|
|
||||||
file->atime = isodate7(data);
|
|
||||||
data += 7;
|
|
||||||
}
|
|
||||||
if (flag & 8) { /* Attribute time. */
|
|
||||||
file->ctime = isodate7(data);
|
|
||||||
data += 7;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
/* FALLTHROUGH */
|
/* FALLTHROUGH */
|
||||||
default:
|
default:
|
||||||
/* The FALLTHROUGHs above leave us here for
|
/* The FALLTHROUGHs above leave us here for
|
||||||
* any unsupported extension. */
|
* any unsupported extension. */
|
||||||
#ifdef DEBUG
|
#if DEBUG
|
||||||
{
|
{
|
||||||
const unsigned char *t;
|
const unsigned char *t;
|
||||||
fprintf(stderr, "\nUnsupported RRIP extension for %s\n", file->name);
|
fprintf(stderr, "\nUnsupported RRIP extension for %s\n", file->name.s);
|
||||||
fprintf(stderr, " %c%c(%d):", p[0], p[1], data_length);
|
fprintf(stderr, " %c%c(%d):", p[0], p[1], data_length);
|
||||||
for (t = data; t < data + data_length && t < data + 16; t++)
|
for (t = data; t < data + data_length && t < data + 16; t++)
|
||||||
fprintf(stderr, " %02x", *t);
|
fprintf(stderr, " %02x", *t);
|
||||||
@ -877,6 +867,215 @@ parse_rockridge(struct iso9660 *iso9660, struct file_info *file,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
parse_rockridge_NM1(struct file_info *file, const unsigned char *data,
|
||||||
|
int data_length)
|
||||||
|
{
|
||||||
|
if (!file->name_continues)
|
||||||
|
archive_string_empty(&file->name);
|
||||||
|
file->name_continues = 0;
|
||||||
|
if (data_length < 1)
|
||||||
|
return;
|
||||||
|
/*
|
||||||
|
* NM version 1 extension comprises:
|
||||||
|
* 1 byte flag, value is one of:
|
||||||
|
* = 0: remainder is name
|
||||||
|
* = 1: remainder is name, next NM entry continues name
|
||||||
|
* = 2: "."
|
||||||
|
* = 4: ".."
|
||||||
|
* = 32: Implementation specific
|
||||||
|
* All other values are reserved.
|
||||||
|
*/
|
||||||
|
switch(data[0]) {
|
||||||
|
case 0:
|
||||||
|
if (data_length < 2)
|
||||||
|
return;
|
||||||
|
archive_strncat(&file->name, data + 1, data_length - 1);
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
if (data_length < 2)
|
||||||
|
return;
|
||||||
|
archive_strncat(&file->name, data + 1, data_length - 1);
|
||||||
|
file->name_continues = 1;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
archive_strcat(&file->name, ".");
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
archive_strcat(&file->name, "..");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
parse_rockridge_TF1(struct file_info *file, const unsigned char *data,
|
||||||
|
int data_length)
|
||||||
|
{
|
||||||
|
char flag;
|
||||||
|
/*
|
||||||
|
* TF extension comprises:
|
||||||
|
* one byte flag
|
||||||
|
* create time (optional)
|
||||||
|
* modify time (optional)
|
||||||
|
* access time (optional)
|
||||||
|
* attribute time (optional)
|
||||||
|
* Time format and presence of fields
|
||||||
|
* is controlled by flag bits.
|
||||||
|
*/
|
||||||
|
if (data_length < 1)
|
||||||
|
return;
|
||||||
|
flag = data[0];
|
||||||
|
++data;
|
||||||
|
--data_length;
|
||||||
|
if (flag & 0x80) {
|
||||||
|
/* Use 17-byte time format. */
|
||||||
|
if ((flag & 1) && data_length >= 17) {
|
||||||
|
/* Create time. */
|
||||||
|
file->birthtime = isodate17(data);
|
||||||
|
data += 17;
|
||||||
|
data_length -= 17;
|
||||||
|
}
|
||||||
|
if ((flag & 2) && data_length >= 17) {
|
||||||
|
/* Modify time. */
|
||||||
|
file->mtime = isodate17(data);
|
||||||
|
data += 17;
|
||||||
|
data_length -= 17;
|
||||||
|
}
|
||||||
|
if ((flag & 4) && data_length >= 17) {
|
||||||
|
/* Access time. */
|
||||||
|
file->atime = isodate17(data);
|
||||||
|
data += 17;
|
||||||
|
data_length -= 17;
|
||||||
|
}
|
||||||
|
if ((flag & 8) && data_length >= 17) {
|
||||||
|
/* Attribute change time. */
|
||||||
|
file->ctime = isodate17(data);
|
||||||
|
data += 17;
|
||||||
|
data_length -= 17;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* Use 7-byte time format. */
|
||||||
|
if ((flag & 1) && data_length >= 7) {
|
||||||
|
/* Create time. */
|
||||||
|
file->birthtime = isodate17(data);
|
||||||
|
data += 7;
|
||||||
|
data_length -= 7;
|
||||||
|
}
|
||||||
|
if ((flag & 2) && data_length >= 7) {
|
||||||
|
/* Modify time. */
|
||||||
|
file->mtime = isodate7(data);
|
||||||
|
data += 7;
|
||||||
|
data_length -= 7;
|
||||||
|
}
|
||||||
|
if ((flag & 4) && data_length >= 7) {
|
||||||
|
/* Access time. */
|
||||||
|
file->atime = isodate7(data);
|
||||||
|
data += 7;
|
||||||
|
data_length -= 7;
|
||||||
|
}
|
||||||
|
if ((flag & 8) && data_length >= 7) {
|
||||||
|
/* Attribute change time. */
|
||||||
|
file->ctime = isodate7(data);
|
||||||
|
data += 7;
|
||||||
|
data_length -= 7;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
parse_rockridge_SL1(struct file_info *file, const unsigned char *data,
|
||||||
|
int data_length)
|
||||||
|
{
|
||||||
|
int component_continues = 1;
|
||||||
|
|
||||||
|
if (!file->symlink_continues)
|
||||||
|
archive_string_empty(&file->symlink);
|
||||||
|
else
|
||||||
|
archive_strcat(&file->symlink, "/");
|
||||||
|
file->symlink_continues = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Defined flag values:
|
||||||
|
* 0: This is the last SL record for this symbolic link
|
||||||
|
* 1: this symbolic link field continues in next SL entry
|
||||||
|
* All other values are reserved.
|
||||||
|
*/
|
||||||
|
if (data_length < 1)
|
||||||
|
return;
|
||||||
|
switch(*data) {
|
||||||
|
case 0:
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
file->symlink_continues = 1;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
++data; /* Skip flag byte. */
|
||||||
|
--data_length;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* SL extension body stores "components".
|
||||||
|
* Basically, this is a complicated way of storing
|
||||||
|
* a POSIX path. It also interferes with using
|
||||||
|
* symlinks for storing non-path data. <sigh>
|
||||||
|
*
|
||||||
|
* Each component is 2 bytes (flag and length)
|
||||||
|
* possibly followed by name data.
|
||||||
|
*/
|
||||||
|
while (data_length >= 2) {
|
||||||
|
unsigned char flag = *data++;
|
||||||
|
unsigned char nlen = *data++;
|
||||||
|
data_length -= 2;
|
||||||
|
|
||||||
|
if (!component_continues)
|
||||||
|
archive_strcat(&file->symlink, "/");
|
||||||
|
component_continues = 0;
|
||||||
|
|
||||||
|
switch(flag) {
|
||||||
|
case 0: /* Usual case, this is text. */
|
||||||
|
if (data_length < nlen)
|
||||||
|
return;
|
||||||
|
archive_strncat(&file->symlink,
|
||||||
|
(const char *)data, nlen);
|
||||||
|
break;
|
||||||
|
case 0x01: /* Text continues in next component. */
|
||||||
|
if (data_length < nlen)
|
||||||
|
return;
|
||||||
|
archive_strncat(&file->symlink,
|
||||||
|
(const char *)data, nlen);
|
||||||
|
component_continues = 1;
|
||||||
|
break;
|
||||||
|
case 0x02: /* Current dir. */
|
||||||
|
archive_strcat(&file->symlink, ".");
|
||||||
|
break;
|
||||||
|
case 0x04: /* Parent dir. */
|
||||||
|
archive_strcat(&file->symlink, "..");
|
||||||
|
break;
|
||||||
|
case 0x08: /* Root of filesystem. */
|
||||||
|
archive_string_empty(&file->symlink);
|
||||||
|
archive_strcat(&file->symlink, "/");
|
||||||
|
break;
|
||||||
|
case 0x10: /* Undefined (historically "volume root" */
|
||||||
|
archive_string_empty(&file->symlink);
|
||||||
|
archive_strcat(&file->symlink, "ROOT");
|
||||||
|
break;
|
||||||
|
case 0x20: /* Undefined (historically "hostname") */
|
||||||
|
archive_strcat(&file->symlink, "hostname");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* TODO: issue a warning ? */
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
data += nlen;
|
||||||
|
data_length -= nlen;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
release_file(struct iso9660 *iso9660, struct file_info *file)
|
release_file(struct iso9660 *iso9660, struct file_info *file)
|
||||||
{
|
{
|
||||||
@ -884,8 +1083,7 @@ release_file(struct iso9660 *iso9660, struct file_info *file)
|
|||||||
|
|
||||||
if (file->refcount == 0) {
|
if (file->refcount == 0) {
|
||||||
parent = file->parent;
|
parent = file->parent;
|
||||||
if (file->name)
|
archive_string_free(&file->name);
|
||||||
free(file->name);
|
|
||||||
archive_string_free(&file->symlink);
|
archive_string_free(&file->symlink);
|
||||||
free(file);
|
free(file);
|
||||||
if (parent != NULL) {
|
if (parent != NULL) {
|
||||||
@ -1074,18 +1272,18 @@ time_from_tm(struct tm *t)
|
|||||||
static const char *
|
static const char *
|
||||||
build_pathname(struct archive_string *as, struct file_info *file)
|
build_pathname(struct archive_string *as, struct file_info *file)
|
||||||
{
|
{
|
||||||
if (file->parent != NULL && file->parent->name[0] != '\0') {
|
if (file->parent != NULL && archive_strlen(&file->parent->name) > 0) {
|
||||||
build_pathname(as, file->parent);
|
build_pathname(as, file->parent);
|
||||||
archive_strcat(as, "/");
|
archive_strcat(as, "/");
|
||||||
}
|
}
|
||||||
if (file->name[0] == '\0')
|
if (archive_strlen(&file->name) == 0)
|
||||||
archive_strcat(as, ".");
|
archive_strcat(as, ".");
|
||||||
else
|
else
|
||||||
archive_strcat(as, file->name);
|
archive_string_concat(as, &file->name);
|
||||||
return (as->s);
|
return (as->s);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG
|
#if DEBUG
|
||||||
static void
|
static void
|
||||||
dump_isodirrec(FILE *out, const unsigned char *isodirrec)
|
dump_isodirrec(FILE *out, const unsigned char *isodirrec)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user