309 lines
8.3 KiB
C
309 lines
8.3 KiB
C
/*
|
|
* Copyright (c) 2007 Vreixo Formoso
|
|
* Copyright (c) 2009 - 2015 Thomas Schmitt
|
|
*
|
|
* 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
|
|
* or later as published by the Free Software Foundation.
|
|
* See COPYING file for details.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "../config.h"
|
|
#endif
|
|
|
|
/* libisofs.h defines aaip_xinfo_func */
|
|
#include "libisofs.h"
|
|
|
|
#include "builder.h"
|
|
#include "node.h"
|
|
#include "fsource.h"
|
|
#include "image.h"
|
|
#include "aaip_0_2.h"
|
|
#include "util.h"
|
|
#include "messages.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
void iso_node_builder_ref(IsoNodeBuilder *builder)
|
|
{
|
|
++builder->refcount;
|
|
}
|
|
|
|
void iso_node_builder_unref(IsoNodeBuilder *builder)
|
|
{
|
|
if (--builder->refcount == 0) {
|
|
/* free private data */
|
|
builder->free(builder);
|
|
free(builder);
|
|
}
|
|
}
|
|
|
|
static
|
|
int default_create_file(IsoNodeBuilder *builder, IsoImage *image,
|
|
IsoFileSource *src, IsoFile **file)
|
|
{
|
|
int ret;
|
|
struct stat info;
|
|
IsoStream *stream;
|
|
IsoFile *node;
|
|
char *name;
|
|
|
|
if (builder == NULL || src == NULL || file == NULL) {
|
|
return ISO_NULL_POINTER;
|
|
}
|
|
|
|
ret = iso_file_source_stat(src, &info);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
/* this will fail if src is a dir, is not accessible... */
|
|
ret = iso_file_source_stream_new(src, &stream);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
/* take a ref to the src, as stream has taken our ref */
|
|
iso_file_source_ref(src);
|
|
|
|
name = iso_file_source_get_name(src);
|
|
if ((int) strlen(name) > image->truncate_length) {
|
|
ret = iso_truncate_rr_name(image->truncate_mode,
|
|
image->truncate_length, name, 0);
|
|
if (ret < 0) {
|
|
iso_stream_unref(stream);
|
|
free(name);
|
|
return ret;
|
|
}
|
|
}
|
|
ret = iso_node_new_file(name, stream, &node);
|
|
if (ret < 0) {
|
|
iso_stream_unref(stream);
|
|
free(name);
|
|
return ret;
|
|
}
|
|
|
|
/* fill node fields */
|
|
iso_node_set_permissions((IsoNode*)node, info.st_mode);
|
|
iso_node_set_uid((IsoNode*)node, info.st_uid);
|
|
iso_node_set_gid((IsoNode*)node, info.st_gid);
|
|
iso_node_set_atime((IsoNode*)node, info.st_atime);
|
|
iso_node_set_mtime((IsoNode*)node, info.st_mtime);
|
|
iso_node_set_ctime((IsoNode*)node, info.st_ctime);
|
|
iso_node_set_uid((IsoNode*)node, info.st_uid);
|
|
|
|
*file = node;
|
|
return ISO_SUCCESS;
|
|
}
|
|
|
|
static
|
|
int default_create_node(IsoNodeBuilder *builder, IsoImage *image,
|
|
IsoFileSource *src, char *in_name, IsoNode **node)
|
|
{
|
|
int ret, name_is_attached = 0;
|
|
struct stat info;
|
|
IsoNode *new;
|
|
IsoFilesystem *fs;
|
|
char *name = NULL;
|
|
unsigned char *aa_string = NULL;
|
|
char *a_text = NULL, *d_text = NULL;
|
|
char *dest = NULL;
|
|
IsoSymlink *link;
|
|
|
|
if (builder == NULL || src == NULL || node == NULL) {
|
|
{ret = ISO_NULL_POINTER; goto ex;}
|
|
}
|
|
|
|
/* get info about source */
|
|
if (iso_tree_get_follow_symlinks(image)) {
|
|
ret = iso_file_source_stat(src, &info);
|
|
} else {
|
|
ret = iso_file_source_lstat(src, &info);
|
|
}
|
|
if (ret < 0) {
|
|
goto ex;
|
|
}
|
|
|
|
if (in_name == NULL) {
|
|
name = iso_file_source_get_name(src);
|
|
} else {
|
|
name = strdup(in_name);
|
|
if (name == NULL) {
|
|
ret = ISO_OUT_OF_MEM; goto ex;
|
|
}
|
|
}
|
|
|
|
if ((int) strlen(name) > image->truncate_length) {
|
|
ret = iso_truncate_rr_name(image->truncate_mode,
|
|
image->truncate_length, name, 0);
|
|
if (ret < 0)
|
|
goto ex;
|
|
}
|
|
fs = iso_file_source_get_filesystem(src);
|
|
new = NULL;
|
|
|
|
switch (info.st_mode & S_IFMT) {
|
|
case S_IFREG:
|
|
{
|
|
/* source is a regular file */
|
|
IsoStream *stream;
|
|
IsoFile *file;
|
|
ret = iso_file_source_stream_new(src, &stream);
|
|
if (ret < 0) {
|
|
break;
|
|
}
|
|
/* take a ref to the src, as stream has taken our ref */
|
|
iso_file_source_ref(src);
|
|
|
|
/* create the file */
|
|
ret = iso_node_new_file(name, stream, &file);
|
|
if (ret < 0) {
|
|
iso_stream_unref(stream);
|
|
}
|
|
new = (IsoNode*) file;
|
|
}
|
|
break;
|
|
case S_IFDIR:
|
|
{
|
|
/* source is a directory */
|
|
IsoDir *dir;
|
|
ret = iso_node_new_dir(name, &dir);
|
|
new = (IsoNode*)dir;
|
|
}
|
|
break;
|
|
case S_IFLNK:
|
|
{
|
|
/* source is a symbolic link */
|
|
LIBISO_ALLOC_MEM(dest, char, LIBISOFS_NODE_PATH_MAX);
|
|
ret = iso_file_source_readlink(src, dest, LIBISOFS_NODE_PATH_MAX);
|
|
if (ret < 0) {
|
|
break;
|
|
}
|
|
ret = iso_node_new_symlink(name, strdup(dest), &link);
|
|
new = (IsoNode*) link;
|
|
if (fs != NULL) {
|
|
link->fs_id = fs->get_id(fs);
|
|
if (link->fs_id != 0) {
|
|
link->st_ino = info.st_ino;
|
|
link->st_dev = info.st_dev;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case S_IFSOCK:
|
|
case S_IFBLK:
|
|
case S_IFCHR:
|
|
case S_IFIFO:
|
|
{
|
|
/* source is an special file */
|
|
IsoSpecial *special;
|
|
ret = iso_node_new_special(name, info.st_mode, info.st_rdev,
|
|
&special);
|
|
new = (IsoNode*) special;
|
|
if (fs != NULL) {
|
|
special->fs_id = fs->get_id(fs);
|
|
if (special->fs_id != 0) {
|
|
special->st_ino = info.st_ino;
|
|
special->st_dev = info.st_dev;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
ret = ISO_BAD_FSRC_FILETYPE;
|
|
goto ex;
|
|
}
|
|
|
|
if (ret < 0)
|
|
goto ex;
|
|
name_is_attached = 1;
|
|
|
|
/* fill fields */
|
|
iso_node_set_perms_internal(new, info.st_mode, 1);
|
|
iso_node_set_uid(new, info.st_uid);
|
|
iso_node_set_gid(new, info.st_gid);
|
|
iso_node_set_atime(new, info.st_atime);
|
|
iso_node_set_mtime(new, info.st_mtime);
|
|
iso_node_set_ctime(new, info.st_ctime);
|
|
iso_node_set_uid(new, info.st_uid);
|
|
|
|
/* Eventually set S_IRWXG from ACL */
|
|
if (image->builder_ignore_acl) {
|
|
ret = iso_file_source_get_aa_string(src, &aa_string, 4);
|
|
if (ret >= 0 && aa_string != NULL)
|
|
iso_aa_get_acl_text(aa_string, info.st_mode, &a_text, &d_text, 16);
|
|
if (ret >= 0 && a_text != NULL) {
|
|
aaip_cleanout_st_mode(a_text, &(info.st_mode), 4 | 16);
|
|
iso_node_set_perms_internal(new, info.st_mode, 1);
|
|
}
|
|
iso_aa_get_acl_text(aa_string, info.st_mode, &a_text, &d_text,
|
|
1 << 15); /* free ACL texts */
|
|
if(aa_string != NULL)
|
|
free(aa_string);
|
|
aa_string = NULL;
|
|
}
|
|
|
|
/* Obtain ownership of eventual AAIP string */
|
|
ret = iso_file_source_get_aa_string(src, &aa_string,
|
|
1 | (image->builder_ignore_acl << 1) |
|
|
(image->builder_ignore_ea << 2) |
|
|
(image->builder_take_all_ea << 3));
|
|
if(ret == 2)
|
|
image->blind_on_local_get_attrs = 1;
|
|
if (ret > 0 && aa_string != NULL) {
|
|
ret = iso_node_add_xinfo(new, aaip_xinfo_func, aa_string);
|
|
if (ret < 0)
|
|
goto ex;
|
|
} else if(aa_string != NULL) {
|
|
free(aa_string);
|
|
}
|
|
|
|
*node = new;
|
|
|
|
ret = ISO_SUCCESS;
|
|
ex:;
|
|
if (name != NULL && !name_is_attached)
|
|
free(name);
|
|
LIBISO_FREE_MEM(dest);
|
|
return ret;
|
|
}
|
|
|
|
static
|
|
void default_free(IsoNodeBuilder *builder)
|
|
{
|
|
/* The .free() method of IsoNodeBuilder shall free private data but not
|
|
the builder itself. The latter is done in iso_node_builder_unref().
|
|
*/
|
|
return;
|
|
}
|
|
|
|
int iso_node_basic_builder_new(IsoNodeBuilder **builder)
|
|
{
|
|
IsoNodeBuilder *b;
|
|
|
|
if (builder == NULL) {
|
|
return ISO_NULL_POINTER;
|
|
}
|
|
|
|
b = malloc(sizeof(IsoNodeBuilder));
|
|
if (b == NULL) {
|
|
return ISO_OUT_OF_MEM;
|
|
}
|
|
|
|
b->refcount = 1;
|
|
b->create_file_data = NULL;
|
|
b->create_node_data = NULL;
|
|
b->create_file = default_create_file;
|
|
b->create_node = default_create_node;
|
|
b->free = default_free;
|
|
|
|
*builder = b;
|
|
return ISO_SUCCESS;
|
|
}
|