Add support for ISO-9660:1999.
This commit is contained in:
parent
46507e68aa
commit
5ed68d20e9
@ -49,6 +49,8 @@ src_libisofs_la_SOURCES = \
|
|||||||
src/joliet.c \
|
src/joliet.c \
|
||||||
src/eltorito.h \
|
src/eltorito.h \
|
||||||
src/eltorito.c \
|
src/eltorito.c \
|
||||||
|
src/iso1999.h \
|
||||||
|
src/iso1999.c \
|
||||||
src/data_source.c
|
src/data_source.c
|
||||||
libinclude_HEADERS = \
|
libinclude_HEADERS = \
|
||||||
src/libisofs.h
|
src/libisofs.h
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <err.h>
|
#include <err.h>
|
||||||
|
|
||||||
const char * const optstring = "JRL:b:hV:";
|
const char * const optstring = "JRIL:b:hV:";
|
||||||
extern char *optarg;
|
extern char *optarg;
|
||||||
extern int optind;
|
extern int optind;
|
||||||
|
|
||||||
@ -31,6 +31,7 @@ void help()
|
|||||||
"Options:\n"
|
"Options:\n"
|
||||||
" -J Add Joliet support\n"
|
" -J Add Joliet support\n"
|
||||||
" -R Add Rock Ridge support\n"
|
" -R Add Rock Ridge support\n"
|
||||||
|
" -I Add ISO 9660:1999 support\n"
|
||||||
" -V label Volume Label\n"
|
" -V label Volume Label\n"
|
||||||
" -L <num> Set the ISO level (1 or 2)\n"
|
" -L <num> Set the ISO level (1 or 2)\n"
|
||||||
" -b file Specifies a boot image to add to image\n"
|
" -b file Specifies a boot image to add to image\n"
|
||||||
@ -53,6 +54,7 @@ int main(int argc, char **argv)
|
|||||||
1, /* level */
|
1, /* level */
|
||||||
0, /* rockridge */
|
0, /* rockridge */
|
||||||
0, /* joliet */
|
0, /* joliet */
|
||||||
|
0, /* iso1999 */
|
||||||
0, /* omit_version_numbers */
|
0, /* omit_version_numbers */
|
||||||
0, /* allow_deep_paths */
|
0, /* allow_deep_paths */
|
||||||
0, /* allow_longer_paths */
|
0, /* allow_longer_paths */
|
||||||
@ -90,6 +92,9 @@ int main(int argc, char **argv)
|
|||||||
case 'R':
|
case 'R':
|
||||||
opts.rockridge = 1;
|
opts.rockridge = 1;
|
||||||
break;
|
break;
|
||||||
|
case 'I':
|
||||||
|
opts.iso1999 = 1;
|
||||||
|
break;
|
||||||
case 'L':
|
case 'L':
|
||||||
opts.level = atoi(optarg);
|
opts.level = atoi(optarg);
|
||||||
break;
|
break;
|
||||||
|
@ -36,6 +36,7 @@ int main(int argc, char **argv)
|
|||||||
1, /* level */
|
1, /* level */
|
||||||
1, /* rockridge */
|
1, /* rockridge */
|
||||||
0, /* joliet */
|
0, /* joliet */
|
||||||
|
0, /* iso1999 */
|
||||||
0, /* omit_version_numbers */
|
0, /* omit_version_numbers */
|
||||||
0, /* allow_deep_paths */
|
0, /* allow_deep_paths */
|
||||||
0, /* allow_longer_paths */
|
0, /* allow_longer_paths */
|
||||||
|
@ -31,6 +31,7 @@ int main(int argc, char **argv)
|
|||||||
1, /* level */
|
1, /* level */
|
||||||
1, /* rockridge */
|
1, /* rockridge */
|
||||||
0, /* joliet */
|
0, /* joliet */
|
||||||
|
0, /* iso1999 */
|
||||||
0, /* omit_version_numbers */
|
0, /* omit_version_numbers */
|
||||||
0, /* allow_deep_paths */
|
0, /* allow_deep_paths */
|
||||||
0, /* allow_longer_paths */
|
0, /* allow_longer_paths */
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
#include "libisofs.h"
|
#include "libisofs.h"
|
||||||
#include "ecma119.h"
|
#include "ecma119.h"
|
||||||
#include "joliet.h"
|
#include "joliet.h"
|
||||||
|
#include "iso1999.h"
|
||||||
#include "eltorito.h"
|
#include "eltorito.h"
|
||||||
#include "ecma119_tree.h"
|
#include "ecma119_tree.h"
|
||||||
#include "error.h"
|
#include "error.h"
|
||||||
@ -291,27 +292,6 @@ void write_one_dir_record(Ecma119Image *t, Ecma119Node *node, int file_id,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Copy up to \p max characters from \p src to \p dest. If \p src has less than
|
|
||||||
* \p max characters, we pad dest with " " characters.
|
|
||||||
*/
|
|
||||||
static
|
|
||||||
void strncpy_pad(char *dest, const char *src, size_t max)
|
|
||||||
{
|
|
||||||
size_t len, i;
|
|
||||||
|
|
||||||
if (src != NULL) {
|
|
||||||
len = MIN(strlen(src), max);
|
|
||||||
} else {
|
|
||||||
len = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i = 0; i < len; ++i)
|
|
||||||
dest[i] = src[i];
|
|
||||||
for (i = len; i < max; ++i)
|
|
||||||
dest[i] = ' ';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Write the Primary Volume Descriptor (ECMA-119, 8.4)
|
* Write the Primary Volume Descriptor (ECMA-119, 8.4)
|
||||||
*/
|
*/
|
||||||
@ -843,6 +823,7 @@ int ecma119_image_new(IsoImage *src, Ecma119WriteOpts *opts, Ecma119Image **img)
|
|||||||
target->iso_level = opts->level;
|
target->iso_level = opts->level;
|
||||||
target->rockridge = opts->rockridge;
|
target->rockridge = opts->rockridge;
|
||||||
target->joliet = opts->joliet;
|
target->joliet = opts->joliet;
|
||||||
|
target->iso1999 = opts->iso1999;
|
||||||
target->ino = 0;
|
target->ino = 0;
|
||||||
target->omit_version_numbers = opts->omit_version_numbers
|
target->omit_version_numbers = opts->omit_version_numbers
|
||||||
| opts->max_37_char_filenames;
|
| opts->max_37_char_filenames;
|
||||||
@ -912,6 +893,9 @@ int ecma119_image_new(IsoImage *src, Ecma119WriteOpts *opts, Ecma119Image **img)
|
|||||||
if (target->joliet) {
|
if (target->joliet) {
|
||||||
nwriters++;
|
nwriters++;
|
||||||
}
|
}
|
||||||
|
if (target->iso1999) {
|
||||||
|
nwriters++;
|
||||||
|
}
|
||||||
|
|
||||||
target->writers = malloc(nwriters * sizeof(void*));
|
target->writers = malloc(nwriters * sizeof(void*));
|
||||||
if (target->writers == NULL) {
|
if (target->writers == NULL) {
|
||||||
@ -941,6 +925,14 @@ int ecma119_image_new(IsoImage *src, Ecma119WriteOpts *opts, Ecma119Image **img)
|
|||||||
goto target_cleanup;
|
goto target_cleanup;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* create writer for ISO 9660:1999 structure */
|
||||||
|
if (target->iso1999) {
|
||||||
|
ret = iso1999_writer_create(target);
|
||||||
|
if (ret < 0) {
|
||||||
|
goto target_cleanup;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
voldesc_size = target->curblock - target->ms_block - 16;
|
voldesc_size = target->curblock - target->ms_block - 16;
|
||||||
|
|
||||||
|
@ -21,6 +21,7 @@
|
|||||||
typedef struct ecma119_image Ecma119Image;
|
typedef struct ecma119_image Ecma119Image;
|
||||||
typedef struct ecma119_node Ecma119Node;
|
typedef struct ecma119_node Ecma119Node;
|
||||||
typedef struct joliet_node JolietNode;
|
typedef struct joliet_node JolietNode;
|
||||||
|
typedef struct iso1999_node Iso1999Node;
|
||||||
typedef struct Iso_File_Src IsoFileSrc;
|
typedef struct Iso_File_Src IsoFileSrc;
|
||||||
typedef struct Iso_Image_Writer IsoImageWriter;
|
typedef struct Iso_Image_Writer IsoImageWriter;
|
||||||
|
|
||||||
@ -35,6 +36,7 @@ struct ecma119_image
|
|||||||
unsigned int rockridge :1;
|
unsigned int rockridge :1;
|
||||||
unsigned int joliet :1;
|
unsigned int joliet :1;
|
||||||
unsigned int eltorito :1;
|
unsigned int eltorito :1;
|
||||||
|
unsigned int iso1999 :1;
|
||||||
|
|
||||||
/* relaxed constraints */
|
/* relaxed constraints */
|
||||||
unsigned int omit_version_numbers :1;
|
unsigned int omit_version_numbers :1;
|
||||||
@ -112,6 +114,15 @@ struct ecma119_image
|
|||||||
uint32_t joliet_l_path_table_pos;
|
uint32_t joliet_l_path_table_pos;
|
||||||
uint32_t joliet_m_path_table_pos;
|
uint32_t joliet_m_path_table_pos;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ISO 9660:1999 related information
|
||||||
|
*/
|
||||||
|
Iso1999Node *iso1999_root;
|
||||||
|
size_t iso1999_ndirs;
|
||||||
|
uint32_t iso1999_path_table_size;
|
||||||
|
uint32_t iso1999_l_path_table_pos;
|
||||||
|
uint32_t iso1999_m_path_table_pos;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* El-Torito related information
|
* El-Torito related information
|
||||||
*/
|
*/
|
||||||
|
799
src/iso1999.c
Normal file
799
src/iso1999.c
Normal file
@ -0,0 +1,799 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2007 Vreixo Formoso
|
||||||
|
*
|
||||||
|
* This file is part of the libisofs project; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License version 2 as
|
||||||
|
* published by the Free Software Foundation. See COPYING file for details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "iso1999.h"
|
||||||
|
#include "messages.h"
|
||||||
|
#include "writer.h"
|
||||||
|
#include "error.h"
|
||||||
|
#include "image.h"
|
||||||
|
#include "filesrc.h"
|
||||||
|
#include "eltorito.h"
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
static
|
||||||
|
char *get_iso1999_name(Ecma119Image *t, const char *str)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
char *name;
|
||||||
|
|
||||||
|
if (str == NULL) {
|
||||||
|
/* not an error, can be root node */
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!strcmp(t->input_charset, t->output_charset)) {
|
||||||
|
/* no conversion needed */
|
||||||
|
name = strdup(str);
|
||||||
|
} else {
|
||||||
|
ret = strconv(str, t->input_charset, t->output_charset, &name);
|
||||||
|
if (ret < 0) {
|
||||||
|
iso_msg_sorry(t->image->messenger, LIBISO_CHARSET_ERROR,
|
||||||
|
"Charset conversion error. Can't convert %s from %s to %s",
|
||||||
|
str, t->input_charset, t->output_charset);
|
||||||
|
|
||||||
|
/* use the original name, it's the best we can do */
|
||||||
|
name = strdup(str);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ISO 9660:1999 7.5.1 */
|
||||||
|
if (strlen(name) > 207) {
|
||||||
|
name[207] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
void iso1999_node_free(Iso1999Node *node)
|
||||||
|
{
|
||||||
|
if (node == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (node->type == ISO1999_DIR) {
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < node->info.dir->nchildren; i++) {
|
||||||
|
iso1999_node_free(node->info.dir->children[i]);
|
||||||
|
}
|
||||||
|
free(node->info.dir->children);
|
||||||
|
free(node->info.dir);
|
||||||
|
}
|
||||||
|
iso_node_unref(node->node);
|
||||||
|
free(node->name);
|
||||||
|
free(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a low level ISO 9660:1999 node
|
||||||
|
* @return
|
||||||
|
* 1 success, 0 ignored, < 0 error
|
||||||
|
*/
|
||||||
|
static
|
||||||
|
int create_node(Ecma119Image *t, IsoNode *iso, Iso1999Node **node)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
Iso1999Node *n;
|
||||||
|
|
||||||
|
n = calloc(1, sizeof(Iso1999Node));
|
||||||
|
if (n == NULL) {
|
||||||
|
return ISO_MEM_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (iso->type == LIBISO_DIR) {
|
||||||
|
IsoDir *dir = (IsoDir*) iso;
|
||||||
|
n->info.dir = calloc(1, sizeof(struct iso1999_dir_info));
|
||||||
|
if (n->info.dir == NULL) {
|
||||||
|
free(n);
|
||||||
|
return ISO_MEM_ERROR;
|
||||||
|
}
|
||||||
|
n->info.dir->children = calloc(sizeof(void*), dir->nchildren);
|
||||||
|
if (n->info.dir->children == NULL) {
|
||||||
|
free(n->info.dir);
|
||||||
|
free(n);
|
||||||
|
return ISO_MEM_ERROR;
|
||||||
|
}
|
||||||
|
n->type = ISO1999_DIR;
|
||||||
|
} else if (iso->type == LIBISO_FILE) {
|
||||||
|
/* it's a file */
|
||||||
|
off_t size;
|
||||||
|
IsoFileSrc *src;
|
||||||
|
IsoFile *file = (IsoFile*) iso;
|
||||||
|
|
||||||
|
size = iso_stream_get_size(file->stream);
|
||||||
|
if (size > (off_t)0xffffffff) {
|
||||||
|
iso_msg_note(t->image->messenger, LIBISO_FILE_IGNORED,
|
||||||
|
"File \"%s\" can't be added to image because is "
|
||||||
|
"greater than 4GB", iso->name);
|
||||||
|
free(n);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = iso_file_src_create(t, file, &src);
|
||||||
|
if (ret < 0) {
|
||||||
|
free(n);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
n->info.file = src;
|
||||||
|
n->type = ISO1999_FILE;
|
||||||
|
} else if (iso->type == LIBISO_BOOT) {
|
||||||
|
/* it's a el-torito boot catalog, that we write as a file */
|
||||||
|
IsoFileSrc *src;
|
||||||
|
|
||||||
|
ret = el_torito_catalog_file_src_create(t, &src);
|
||||||
|
if (ret < 0) {
|
||||||
|
free(n);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
n->info.file = src;
|
||||||
|
n->type = ISO1999_FILE;
|
||||||
|
} else {
|
||||||
|
/* should never happen */
|
||||||
|
free(n);
|
||||||
|
return ISO_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* take a ref to the IsoNode */
|
||||||
|
n->node = iso;
|
||||||
|
iso_node_ref(iso);
|
||||||
|
|
||||||
|
*node = n;
|
||||||
|
return ISO_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create the low level ISO 9660:1999 tree from the high level ISO tree.
|
||||||
|
*
|
||||||
|
* @return
|
||||||
|
* 1 success, 0 file ignored, < 0 error
|
||||||
|
*/
|
||||||
|
static
|
||||||
|
int create_tree(Ecma119Image *t, IsoNode *iso, Iso1999Node **tree, int pathlen)
|
||||||
|
{
|
||||||
|
int ret, max_path;
|
||||||
|
Iso1999Node *node = NULL;
|
||||||
|
char *iso_name = NULL;
|
||||||
|
|
||||||
|
if (t == NULL || iso == NULL || tree == NULL) {
|
||||||
|
return ISO_NULL_POINTER;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (iso->hidden & LIBISO_HIDE_ON_1999) {
|
||||||
|
/* file will be ignored */
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
iso_name = get_iso1999_name(t, iso->name);
|
||||||
|
max_path = pathlen + 1 + (iso_name ? strlen(iso_name): 0);
|
||||||
|
if (!t->allow_longer_paths && max_path > 255) {
|
||||||
|
iso_msg_note(t->image->messenger, LIBISO_FILE_IGNORED,
|
||||||
|
"File \"%s\" can't be added to ISO 9660:1999 tree, "
|
||||||
|
"because its path length is larger than 255", iso->name);
|
||||||
|
free(iso_name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (iso->type) {
|
||||||
|
case LIBISO_FILE:
|
||||||
|
ret = create_node(t, iso, &node);
|
||||||
|
break;
|
||||||
|
case LIBISO_DIR:
|
||||||
|
{
|
||||||
|
IsoNode *pos;
|
||||||
|
IsoDir *dir = (IsoDir*)iso;
|
||||||
|
ret = create_node(t, iso, &node);
|
||||||
|
if (ret < 0) {
|
||||||
|
free(iso_name);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
pos = dir->children;
|
||||||
|
while (pos) {
|
||||||
|
int cret;
|
||||||
|
Iso1999Node *child;
|
||||||
|
cret = create_tree(t, pos, &child, max_path);
|
||||||
|
if (cret < 0) {
|
||||||
|
/* error */
|
||||||
|
iso1999_node_free(node);
|
||||||
|
ret = cret;
|
||||||
|
break;
|
||||||
|
} else if (cret == ISO_SUCCESS) {
|
||||||
|
/* add child to this node */
|
||||||
|
int nchildren = node->info.dir->nchildren++;
|
||||||
|
node->info.dir->children[nchildren] = child;
|
||||||
|
child->parent = node;
|
||||||
|
}
|
||||||
|
pos = pos->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case LIBISO_BOOT:
|
||||||
|
if (t->eltorito) {
|
||||||
|
ret = create_node(t, iso, &node);
|
||||||
|
} else {
|
||||||
|
/* log and ignore */
|
||||||
|
iso_msg_note(t->image->messenger, LIBISO_FILE_IGNORED,
|
||||||
|
"El-Torito catalog found on a image without El-Torito.",
|
||||||
|
iso->name);
|
||||||
|
ret = 0;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case LIBISO_SYMLINK:
|
||||||
|
case LIBISO_SPECIAL:
|
||||||
|
iso_msg_note(t->image->messenger, LIBISO_FILE_IGNORED,
|
||||||
|
"Can't add %s to ISO 9660:1999 tree. This kind of files "
|
||||||
|
"can only be added to a Rock Ridget tree. Skipping.",
|
||||||
|
iso->name);
|
||||||
|
ret = 0;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* should never happen */
|
||||||
|
return ISO_ERROR;
|
||||||
|
}
|
||||||
|
if (ret <= 0) {
|
||||||
|
free(iso_name);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
node->name = iso_name;
|
||||||
|
*tree = node;
|
||||||
|
return ISO_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
cmp_node(const void *f1, const void *f2)
|
||||||
|
{
|
||||||
|
Iso1999Node *f = *((Iso1999Node**)f1);
|
||||||
|
Iso1999Node *g = *((Iso1999Node**)f2);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TODO strcmp do not does exactly what ISO 9660:1999, 9.3, as characters
|
||||||
|
* < 0x20 " " are allowed, so name len must be taken into accout
|
||||||
|
*/
|
||||||
|
return strcmp(f->name, g->name);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sort the entries inside an ISO 9660:1999 directory, according to
|
||||||
|
* ISO 9660:1999, 9.3
|
||||||
|
*/
|
||||||
|
static
|
||||||
|
void sort_tree(Iso1999Node *root)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
|
||||||
|
qsort(root->info.dir->children, root->info.dir->nchildren,
|
||||||
|
sizeof(void*), cmp_node);
|
||||||
|
for (i = 0; i < root->info.dir->nchildren; i++) {
|
||||||
|
Iso1999Node *child = root->info.dir->children[i];
|
||||||
|
if (child->type == ISO1999_DIR)
|
||||||
|
sort_tree(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int iso1999_tree_create(Ecma119Image *t)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
Iso1999Node *root;
|
||||||
|
|
||||||
|
if (t == NULL) {
|
||||||
|
return ISO_NULL_POINTER;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = create_tree(t, (IsoNode*)t->image->root, &root, 0);
|
||||||
|
if (ret <= 0) {
|
||||||
|
if (ret == 0) {
|
||||||
|
/* unexpected error, root ignored!! This can't happen */
|
||||||
|
ret = ISO_ERROR;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* the ISO 9660:1999 tree is stored in Ecma119Image target */
|
||||||
|
t->iso1999_root = root;
|
||||||
|
|
||||||
|
iso_msg_debug(t->image->messenger, "Sorting the ISO 9660:1999 tree...");
|
||||||
|
sort_tree(root);
|
||||||
|
|
||||||
|
//iso_msg_debug(t->image->messenger, "Mangling ISO 9660:1999 names...");
|
||||||
|
// FIXME ret = mangle_tree(t, 1);
|
||||||
|
|
||||||
|
return ISO_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compute the size of a directory entry for a single node
|
||||||
|
*/
|
||||||
|
static
|
||||||
|
size_t calc_dirent_len(Ecma119Image *t, Iso1999Node *n)
|
||||||
|
{
|
||||||
|
int ret = n->name ? strlen(n->name) + 33 : 34;
|
||||||
|
if (ret % 2)
|
||||||
|
ret++;
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Computes the total size of all directory entries of a single dir, as
|
||||||
|
* stated in ISO 9660:1999, 6.8.1.3
|
||||||
|
*/
|
||||||
|
static
|
||||||
|
size_t calc_dir_size(Ecma119Image *t, Iso1999Node *dir)
|
||||||
|
{
|
||||||
|
size_t i, len;
|
||||||
|
|
||||||
|
/* size of "." and ".." entries */
|
||||||
|
len = 34 + 34;
|
||||||
|
|
||||||
|
for (i = 0; i < dir->info.dir->nchildren; ++i) {
|
||||||
|
size_t remaining;
|
||||||
|
Iso1999Node *child = dir->info.dir->children[i];
|
||||||
|
size_t dirent_len = calc_dirent_len(t, child);
|
||||||
|
remaining = BLOCK_SIZE - (len % BLOCK_SIZE);
|
||||||
|
if (dirent_len > remaining) {
|
||||||
|
/* child directory entry doesn't fit on block */
|
||||||
|
len += remaining + dirent_len;
|
||||||
|
} else {
|
||||||
|
len += dirent_len;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The size of a dir is always a multiple of block size, as we must add
|
||||||
|
* the size of the unused space after the last directory record
|
||||||
|
* (ISO 9660:1999, 6.8.1.3)
|
||||||
|
*/
|
||||||
|
len = div_up(len, BLOCK_SIZE) * BLOCK_SIZE;
|
||||||
|
|
||||||
|
/* cache the len */
|
||||||
|
dir->info.dir->len = len;
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
void calc_dir_pos(Ecma119Image *t, Iso1999Node *dir)
|
||||||
|
{
|
||||||
|
size_t i, len;
|
||||||
|
|
||||||
|
t->iso1999_ndirs++;
|
||||||
|
dir->info.dir->block = t->curblock;
|
||||||
|
len = calc_dir_size(t, dir);
|
||||||
|
t->curblock += div_up(len, BLOCK_SIZE);
|
||||||
|
for (i = 0; i < dir->info.dir->nchildren; i++) {
|
||||||
|
Iso1999Node *child = dir->info.dir->children[i];
|
||||||
|
if (child->type == ISO1999_DIR) {
|
||||||
|
calc_dir_pos(t, child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compute the length of the path table (ISO 9660:1999, 6.9), in bytes.
|
||||||
|
*/
|
||||||
|
static
|
||||||
|
uint32_t calc_path_table_size(Iso1999Node *dir)
|
||||||
|
{
|
||||||
|
uint32_t size;
|
||||||
|
size_t i;
|
||||||
|
|
||||||
|
/* size of path table for this entry */
|
||||||
|
size = 8;
|
||||||
|
size += dir->name ? strlen(dir->name) : 2;
|
||||||
|
size += (size % 2);
|
||||||
|
|
||||||
|
/* and recurse */
|
||||||
|
for (i = 0; i < dir->info.dir->nchildren; i++) {
|
||||||
|
Iso1999Node *child = dir->info.dir->children[i];
|
||||||
|
if (child->type == ISO1999_DIR) {
|
||||||
|
size += calc_path_table_size(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int iso1999_writer_compute_data_blocks(IsoImageWriter *writer)
|
||||||
|
{
|
||||||
|
Ecma119Image *t;
|
||||||
|
uint32_t path_table_size;
|
||||||
|
|
||||||
|
if (writer == NULL) {
|
||||||
|
return ISO_MEM_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
t = writer->target;
|
||||||
|
|
||||||
|
/* compute position of directories */
|
||||||
|
iso_msg_debug(t->image->messenger,
|
||||||
|
"Computing position of ISO 9660:1999 dir structure");
|
||||||
|
t->iso1999_ndirs = 0;
|
||||||
|
calc_dir_pos(t, t->iso1999_root);
|
||||||
|
|
||||||
|
/* compute length of pathlist */
|
||||||
|
iso_msg_debug(t->image->messenger,
|
||||||
|
"Computing length of ISO 9660:1999 pathlist");
|
||||||
|
path_table_size = calc_path_table_size(t->iso1999_root);
|
||||||
|
|
||||||
|
/* compute location for path tables */
|
||||||
|
t->iso1999_l_path_table_pos = t->curblock;
|
||||||
|
t->curblock += div_up(path_table_size, BLOCK_SIZE);
|
||||||
|
t->iso1999_m_path_table_pos = t->curblock;
|
||||||
|
t->curblock += div_up(path_table_size, BLOCK_SIZE);
|
||||||
|
t->iso1999_path_table_size = path_table_size;
|
||||||
|
|
||||||
|
return ISO_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write a single directory record (ISO 9660:1999, 9.1).
|
||||||
|
*
|
||||||
|
* @param file_id
|
||||||
|
* if >= 0, we use it instead of the filename (for "." and ".." entries).
|
||||||
|
* @param len_fi
|
||||||
|
* Computed length of the file identifier.
|
||||||
|
*/
|
||||||
|
static
|
||||||
|
void write_one_dir_record(Ecma119Image *t, Iso1999Node *node, int file_id,
|
||||||
|
uint8_t *buf, size_t len_fi)
|
||||||
|
{
|
||||||
|
uint32_t len;
|
||||||
|
uint32_t block;
|
||||||
|
uint8_t len_dr; /*< size of dir entry */
|
||||||
|
uint8_t *name = (file_id >= 0) ? (uint8_t*)&file_id
|
||||||
|
: (uint8_t*)node->name;
|
||||||
|
|
||||||
|
struct ecma119_dir_record *rec = (struct ecma119_dir_record*)buf;
|
||||||
|
|
||||||
|
len_dr = 33 + len_fi + (len_fi % 2 ? 0 : 1);
|
||||||
|
|
||||||
|
memcpy(rec->file_id, name, len_fi);
|
||||||
|
|
||||||
|
if (node->type == ISO1999_DIR) {
|
||||||
|
/* use the cached length */
|
||||||
|
len = node->info.dir->len;
|
||||||
|
block = node->info.dir->block;
|
||||||
|
} else if (node->type == ISO1999_FILE) {
|
||||||
|
len = iso_file_src_get_size(node->info.file);
|
||||||
|
block = node->info.file->block;
|
||||||
|
} else {
|
||||||
|
/*
|
||||||
|
* for nodes other than files and dirs, we set both
|
||||||
|
* len and block to 0
|
||||||
|
*/
|
||||||
|
len = 0;
|
||||||
|
block = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* For ".." entry we need to write the parent info!
|
||||||
|
*/
|
||||||
|
if (file_id == 1 && node->parent)
|
||||||
|
node = node->parent;
|
||||||
|
|
||||||
|
rec->len_dr[0] = len_dr;
|
||||||
|
iso_bb(rec->block, block, 4);
|
||||||
|
iso_bb(rec->length, len, 4);
|
||||||
|
iso_datetime_7(rec->recording_time, t->now);
|
||||||
|
rec->flags[0] = (node->type == ISO1999_DIR) ? 2 : 0;
|
||||||
|
iso_bb(rec->vol_seq_number, 1, 2);
|
||||||
|
rec->len_fi[0] = len_fi;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write the enhanced volume descriptor (ISO/IEC 9660:1999, 8.5)
|
||||||
|
*/
|
||||||
|
static
|
||||||
|
int iso1999_writer_write_vol_desc(IsoImageWriter *writer)
|
||||||
|
{
|
||||||
|
IsoImage *image;
|
||||||
|
Ecma119Image *t;
|
||||||
|
|
||||||
|
/* The enhanced volume descriptor is like the sup vol desc */
|
||||||
|
struct ecma119_sup_vol_desc vol;
|
||||||
|
|
||||||
|
char *vol_id = NULL, *pub_id = NULL, *data_id = NULL;
|
||||||
|
char *volset_id = NULL, *system_id = NULL, *application_id = NULL;
|
||||||
|
char *copyright_file_id = NULL, *abstract_file_id = NULL;
|
||||||
|
char *biblio_file_id = NULL;
|
||||||
|
|
||||||
|
if (writer == NULL) {
|
||||||
|
return ISO_MEM_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
t = writer->target;
|
||||||
|
image = t->image;
|
||||||
|
|
||||||
|
iso_msg_debug(image->messenger, "Write Enhanced Vol Desc (ISO 9660:1999)");
|
||||||
|
|
||||||
|
memset(&vol, 0, sizeof(struct ecma119_sup_vol_desc));
|
||||||
|
|
||||||
|
vol_id = get_iso1999_name(t, image->volume_id);
|
||||||
|
str2a_char(t->input_charset, image->publisher_id, &pub_id);
|
||||||
|
str2a_char(t->input_charset, image->data_preparer_id, &data_id);
|
||||||
|
volset_id = get_iso1999_name(t, image->volset_id);
|
||||||
|
|
||||||
|
str2a_char(t->input_charset, image->system_id, &system_id);
|
||||||
|
str2a_char(t->input_charset, image->application_id, &application_id);
|
||||||
|
copyright_file_id = get_iso1999_name(t, image->copyright_file_id);
|
||||||
|
abstract_file_id = get_iso1999_name(t, image->abstract_file_id);
|
||||||
|
biblio_file_id = get_iso1999_name(t, image->biblio_file_id);
|
||||||
|
|
||||||
|
vol.vol_desc_type[0] = 2;
|
||||||
|
memcpy(vol.std_identifier, "CD001", 5);
|
||||||
|
|
||||||
|
/* descriptor version is 2 (ISO/IEC 9660:1999, 8.5.2) */
|
||||||
|
vol.vol_desc_version[0] = 2;
|
||||||
|
strncpy_pad((char*)vol.volume_id, vol_id, 32);
|
||||||
|
|
||||||
|
iso_bb(vol.vol_space_size, t->vol_space_size, 4);
|
||||||
|
iso_bb(vol.vol_set_size, 1, 2);
|
||||||
|
iso_bb(vol.vol_seq_number, 1, 2);
|
||||||
|
iso_bb(vol.block_size, BLOCK_SIZE, 2);
|
||||||
|
iso_bb(vol.path_table_size, t->iso1999_path_table_size, 4);
|
||||||
|
iso_lsb(vol.l_path_table_pos, t->iso1999_l_path_table_pos, 4);
|
||||||
|
iso_msb(vol.m_path_table_pos, t->iso1999_m_path_table_pos, 4);
|
||||||
|
|
||||||
|
write_one_dir_record(t, t->iso1999_root, 0, vol.root_dir_record, 1);
|
||||||
|
|
||||||
|
strncpy_pad((char*)vol.vol_set_id, volset_id, 128);
|
||||||
|
strncpy_pad((char*)vol.publisher_id, pub_id, 128);
|
||||||
|
strncpy_pad((char*)vol.data_prep_id, data_id, 128);
|
||||||
|
|
||||||
|
strncpy_pad((char*)vol.system_id, system_id, 32);
|
||||||
|
|
||||||
|
strncpy_pad((char*)vol.application_id, application_id, 128);
|
||||||
|
strncpy_pad((char*)vol.copyright_file_id, copyright_file_id, 37);
|
||||||
|
strncpy_pad((char*)vol.abstract_file_id, abstract_file_id, 37);
|
||||||
|
strncpy_pad((char*)vol.bibliographic_file_id, biblio_file_id, 37);
|
||||||
|
|
||||||
|
iso_datetime_17(vol.vol_creation_time, t->now);
|
||||||
|
iso_datetime_17(vol.vol_modification_time, t->now);
|
||||||
|
iso_datetime_17(vol.vol_effective_time, t->now);
|
||||||
|
vol.file_structure_version[0] = 1;
|
||||||
|
|
||||||
|
free(vol_id);
|
||||||
|
free(volset_id);
|
||||||
|
free(pub_id);
|
||||||
|
free(data_id);
|
||||||
|
free(system_id);
|
||||||
|
free(application_id);
|
||||||
|
free(copyright_file_id);
|
||||||
|
free(abstract_file_id);
|
||||||
|
free(biblio_file_id);
|
||||||
|
|
||||||
|
/* Finally write the Volume Descriptor */
|
||||||
|
return iso_write(t, &vol, sizeof(struct ecma119_sup_vol_desc));
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int write_one_dir(Ecma119Image *t, Iso1999Node *dir)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
uint8_t buffer[BLOCK_SIZE];
|
||||||
|
size_t i;
|
||||||
|
size_t fi_len, len;
|
||||||
|
|
||||||
|
/* buf will point to current write position on buffer */
|
||||||
|
uint8_t *buf = buffer;
|
||||||
|
|
||||||
|
/* initialize buffer with 0s */
|
||||||
|
memset(buffer, 0, BLOCK_SIZE);
|
||||||
|
|
||||||
|
/* write the "." and ".." entries first */
|
||||||
|
write_one_dir_record(t, dir, 0, buf, 1);
|
||||||
|
buf += 34;
|
||||||
|
write_one_dir_record(t, dir, 1, buf, 1);
|
||||||
|
buf += 34;
|
||||||
|
|
||||||
|
for (i = 0; i < dir->info.dir->nchildren; i++) {
|
||||||
|
Iso1999Node *child = dir->info.dir->children[i];
|
||||||
|
|
||||||
|
/* compute len of directory entry */
|
||||||
|
fi_len = strlen(child->name);
|
||||||
|
len = fi_len + 33 + (fi_len % 2 ? 0 : 1);
|
||||||
|
|
||||||
|
if ( (buf + len - buffer) > BLOCK_SIZE) {
|
||||||
|
/* dir doesn't fit in current block */
|
||||||
|
ret = iso_write(t, buffer, BLOCK_SIZE);
|
||||||
|
if (ret < 0) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
memset(buffer, 0, BLOCK_SIZE);
|
||||||
|
buf = buffer;
|
||||||
|
}
|
||||||
|
/* write the directory entry in any case */
|
||||||
|
write_one_dir_record(t, child, -1, buf, fi_len);
|
||||||
|
buf += len;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* write the last block */
|
||||||
|
ret = iso_write(t, buffer, BLOCK_SIZE);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int write_dirs(Ecma119Image *t, Iso1999Node *root)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
size_t i;
|
||||||
|
|
||||||
|
/* write all directory entries for this dir */
|
||||||
|
ret = write_one_dir(t, root);
|
||||||
|
if (ret < 0) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* recurse */
|
||||||
|
for (i = 0; i < root->info.dir->nchildren; i++) {
|
||||||
|
Iso1999Node *child = root->info.dir->children[i];
|
||||||
|
if (child->type == ISO1999_DIR) {
|
||||||
|
ret = write_dirs(t, child);
|
||||||
|
if (ret < 0) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ISO_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int write_path_table(Ecma119Image *t, Iso1999Node **pathlist, int l_type)
|
||||||
|
{
|
||||||
|
size_t i, len;
|
||||||
|
uint8_t buf[256]; /* 256 is just a convenient size larger enought */
|
||||||
|
struct ecma119_path_table_record *rec;
|
||||||
|
void (*write_int)(uint8_t*, uint32_t, int);
|
||||||
|
Iso1999Node *dir;
|
||||||
|
uint32_t path_table_size;
|
||||||
|
int parent = 0;
|
||||||
|
int ret= ISO_SUCCESS;
|
||||||
|
|
||||||
|
path_table_size = 0;
|
||||||
|
write_int = l_type ? iso_lsb : iso_msb;
|
||||||
|
|
||||||
|
for (i = 0; i < t->iso1999_ndirs; i++) {
|
||||||
|
dir = pathlist[i];
|
||||||
|
|
||||||
|
/* find the index of the parent in the table */
|
||||||
|
while ((i) && pathlist[parent] != dir->parent) {
|
||||||
|
parent++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* write the Path Table Record (ECMA-119, 9.4) */
|
||||||
|
memset(buf, 0, 256);
|
||||||
|
rec = (struct ecma119_path_table_record*) buf;
|
||||||
|
rec->len_di[0] = dir->parent ? (uint8_t) strlen(dir->name) : 1;
|
||||||
|
rec->len_xa[0] = 0;
|
||||||
|
write_int(rec->block, dir->info.dir->block, 4);
|
||||||
|
write_int(rec->parent, parent + 1, 2);
|
||||||
|
if (dir->parent) {
|
||||||
|
memcpy(rec->dir_id, dir->name, rec->len_di[0]);
|
||||||
|
}
|
||||||
|
len = 8 + rec->len_di[0] + (rec->len_di[0] % 2);
|
||||||
|
ret = iso_write(t, buf, len);
|
||||||
|
if (ret < 0) {
|
||||||
|
/* error */
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
path_table_size += len;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* we need to fill the last block with zeros */
|
||||||
|
path_table_size %= BLOCK_SIZE;
|
||||||
|
if (path_table_size) {
|
||||||
|
uint8_t zeros[BLOCK_SIZE];
|
||||||
|
len = BLOCK_SIZE - path_table_size;
|
||||||
|
memset(zeros, 0, len);
|
||||||
|
ret = iso_write(t, zeros, len);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int write_path_tables(Ecma119Image *t)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
size_t i, j, cur;
|
||||||
|
Iso1999Node **pathlist;
|
||||||
|
|
||||||
|
iso_msg_debug(t->image->messenger, "Writing ISO 9660:1999 Path tables");
|
||||||
|
|
||||||
|
/* allocate temporal pathlist */
|
||||||
|
pathlist = malloc(sizeof(void*) * t->iso1999_ndirs);
|
||||||
|
if (pathlist == NULL) {
|
||||||
|
return ISO_MEM_ERROR;
|
||||||
|
}
|
||||||
|
pathlist[0] = t->iso1999_root;
|
||||||
|
cur = 1;
|
||||||
|
|
||||||
|
for (i = 0; i < t->iso1999_ndirs; i++) {
|
||||||
|
Iso1999Node *dir = pathlist[i];
|
||||||
|
for (j = 0; j < dir->info.dir->nchildren; j++) {
|
||||||
|
Iso1999Node *child = dir->info.dir->children[j];
|
||||||
|
if (child->type == ISO1999_DIR) {
|
||||||
|
pathlist[cur++] = child;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Write L Path Table */
|
||||||
|
ret = write_path_table(t, pathlist, 1);
|
||||||
|
if (ret < 0) {
|
||||||
|
goto write_path_tables_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Write L Path Table */
|
||||||
|
ret = write_path_table(t, pathlist, 0);
|
||||||
|
|
||||||
|
write_path_tables_exit: ;
|
||||||
|
free(pathlist);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int iso1999_writer_write_data(IsoImageWriter *writer)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
Ecma119Image *t;
|
||||||
|
|
||||||
|
if (writer == NULL) {
|
||||||
|
return ISO_NULL_POINTER;
|
||||||
|
}
|
||||||
|
t = writer->target;
|
||||||
|
|
||||||
|
/* first of all, we write the directory structure */
|
||||||
|
ret = write_dirs(t, t->iso1999_root);
|
||||||
|
if (ret < 0) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* and write the path tables */
|
||||||
|
ret = write_path_tables(t);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int iso1999_writer_free_data(IsoImageWriter *writer)
|
||||||
|
{
|
||||||
|
/* free the ISO 9660:1999 tree */
|
||||||
|
Ecma119Image *t = writer->target;
|
||||||
|
iso1999_node_free(t->iso1999_root);
|
||||||
|
return ISO_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
int iso1999_writer_create(Ecma119Image *target)
|
||||||
|
{
|
||||||
|
int ret;
|
||||||
|
IsoImageWriter *writer;
|
||||||
|
|
||||||
|
writer = malloc(sizeof(IsoImageWriter));
|
||||||
|
if (writer == NULL) {
|
||||||
|
return ISO_MEM_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
writer->compute_data_blocks = iso1999_writer_compute_data_blocks;
|
||||||
|
writer->write_vol_desc = iso1999_writer_write_vol_desc;
|
||||||
|
writer->write_data = iso1999_writer_write_data;
|
||||||
|
writer->free_data = iso1999_writer_free_data;
|
||||||
|
writer->data = NULL;
|
||||||
|
writer->target = target;
|
||||||
|
|
||||||
|
iso_msg_debug(target->image->messenger,
|
||||||
|
"Creating low level ISO 9660:1999 tree...");
|
||||||
|
ret = iso1999_tree_create(target);
|
||||||
|
if (ret < 0) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add this writer to image */
|
||||||
|
target->writers[target->nwriters++] = writer;
|
||||||
|
|
||||||
|
/* we need the volume descriptor */
|
||||||
|
target->curblock++;
|
||||||
|
return ISO_SUCCESS;
|
||||||
|
}
|
59
src/iso1999.h
Normal file
59
src/iso1999.h
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2007 Vreixo Formoso
|
||||||
|
*
|
||||||
|
* This file is part of the libisofs project; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License version 2 as
|
||||||
|
* published by the Free Software Foundation. See COPYING file for details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Structures related to ISO/IEC 9660:1999, that is version 2 of ISO-9660
|
||||||
|
* "See doc/devel/cookbook/ISO 9660-1999" and
|
||||||
|
* ISO/IEC DIS 9660:1999(E) "Information processing. Volume and file structure
|
||||||
|
* of CD-ROM for Information Interchange"
|
||||||
|
* for further details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef LIBISO_ISO1999_H
|
||||||
|
#define LIBISO_ISO1999_H
|
||||||
|
|
||||||
|
#include "libisofs.h"
|
||||||
|
#include "ecma119.h"
|
||||||
|
|
||||||
|
enum iso1999_node_type {
|
||||||
|
ISO1999_FILE,
|
||||||
|
ISO1999_DIR
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iso1999_dir_info {
|
||||||
|
Iso1999Node **children;
|
||||||
|
size_t nchildren;
|
||||||
|
size_t len;
|
||||||
|
size_t block;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iso1999_node
|
||||||
|
{
|
||||||
|
char *name; /**< Name chosen output charset. */
|
||||||
|
|
||||||
|
Iso1999Node *parent;
|
||||||
|
|
||||||
|
IsoNode *node; /*< reference to the iso node */
|
||||||
|
|
||||||
|
enum iso1999_node_type type;
|
||||||
|
union {
|
||||||
|
IsoFileSrc *file;
|
||||||
|
struct iso1999_dir_info *dir;
|
||||||
|
} info;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a IsoWriter to deal with ISO 9660:1999 estructures, and add it to
|
||||||
|
* the given target.
|
||||||
|
*
|
||||||
|
* @return
|
||||||
|
* 1 on success, < 0 on error
|
||||||
|
*/
|
||||||
|
int iso1999_writer_create(Ecma119Image *target);
|
||||||
|
|
||||||
|
#endif /* LIBISO_ISO1999_H */
|
@ -473,7 +473,6 @@ void write_one_dir_record(Ecma119Image *t, JolietNode *node, int file_id,
|
|||||||
len = iso_file_src_get_size(node->info.file);
|
len = iso_file_src_get_size(node->info.file);
|
||||||
block = node->info.file->block;
|
block = node->info.file->block;
|
||||||
} else {
|
} else {
|
||||||
//TODO el-torito???!?
|
|
||||||
/*
|
/*
|
||||||
* for nodes other than files and dirs, we set both
|
* for nodes other than files and dirs, we set both
|
||||||
* len and block to 0
|
* len and block to 0
|
||||||
|
@ -55,8 +55,12 @@ enum IsoNodeType {
|
|||||||
* \see iso_node_set_hidden
|
* \see iso_node_set_hidden
|
||||||
*/
|
*/
|
||||||
enum IsoHideNodeFlag {
|
enum IsoHideNodeFlag {
|
||||||
|
/** Hide the node in the ECMA-119 / RR tree */
|
||||||
LIBISO_HIDE_ON_RR = 1 << 0,
|
LIBISO_HIDE_ON_RR = 1 << 0,
|
||||||
LIBISO_HIDE_ON_JOLIET = 1 << 1
|
/** Hide the node in the Joliet tree, if Joliet extension are enabled */
|
||||||
|
LIBISO_HIDE_ON_JOLIET = 1 << 1,
|
||||||
|
/** Hide the node in the ISO-9660:1999 tree, if that format is enabled */
|
||||||
|
LIBISO_HIDE_ON_1999 = 1 << 2
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -99,6 +103,7 @@ typedef struct
|
|||||||
/** Which extensions to support. */
|
/** Which extensions to support. */
|
||||||
unsigned int rockridge :1;
|
unsigned int rockridge :1;
|
||||||
unsigned int joliet :1;
|
unsigned int joliet :1;
|
||||||
|
unsigned int iso1999 :1;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Relaxed constraints. Setting any of these to 1 break the specifications,
|
* Relaxed constraints. Setting any of these to 1 break the specifications,
|
||||||
|
20
src/util.c
20
src/util.c
@ -1086,6 +1086,26 @@ char *strcopy(const char *buf, size_t len)
|
|||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copy up to \p max characters from \p src to \p dest. If \p src has less than
|
||||||
|
* \p max characters, we pad dest with " " characters.
|
||||||
|
*/
|
||||||
|
void strncpy_pad(char *dest, const char *src, size_t max)
|
||||||
|
{
|
||||||
|
size_t len, i;
|
||||||
|
|
||||||
|
if (src != NULL) {
|
||||||
|
len = MIN(strlen(src), max);
|
||||||
|
} else {
|
||||||
|
len = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < len; ++i)
|
||||||
|
dest[i] = src[i];
|
||||||
|
for (i = len; i < max; ++i)
|
||||||
|
dest[i] = ' ';
|
||||||
|
}
|
||||||
|
|
||||||
char *ucs2str(const char *buf, size_t len)
|
char *ucs2str(const char *buf, size_t len)
|
||||||
{
|
{
|
||||||
size_t outbytes, inbytes;
|
size_t outbytes, inbytes;
|
||||||
|
@ -230,6 +230,12 @@ int iso_eaccess(const char *path);
|
|||||||
*/
|
*/
|
||||||
char *strcopy(const char *buf, size_t len);
|
char *strcopy(const char *buf, size_t len);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Copy up to \p max characters from \p src to \p dest. If \p src has less than
|
||||||
|
* \p max characters, we pad dest with " " characters.
|
||||||
|
*/
|
||||||
|
void strncpy_pad(char *dest, const char *src, size_t max);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a Joliet string with a length of \p len bytes to a new string
|
* Convert a Joliet string with a length of \p len bytes to a new string
|
||||||
* in local charset.
|
* in local charset.
|
||||||
|
Loading…
Reference in New Issue
Block a user