/*- * Copyright (c) 2013 Marek Kubica * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "archive_platform.h" #ifdef HAVE_ERRNO_H #include <errno.h> #endif #include "archive_entry.h" #include "archive_write_private.h" static ssize_t archive_write_raw_data(struct archive_write *, const void *buff, size_t s); static int archive_write_raw_free(struct archive_write *); static int archive_write_raw_header(struct archive_write *, struct archive_entry *); struct raw { int entries_written; }; /* * Set output format to 'raw' format. */ int archive_write_set_format_raw(struct archive *_a) { struct archive_write *a = (struct archive_write *)_a; struct raw *raw; archive_check_magic(_a, ARCHIVE_WRITE_MAGIC, ARCHIVE_STATE_NEW, "archive_write_set_format_raw"); /* If someone else was already registered, unregister them. */ if (a->format_free != NULL) (a->format_free)(a); raw = (struct raw *)calloc(1, sizeof(*raw)); if (raw == NULL) { archive_set_error(&a->archive, ENOMEM, "Can't allocate raw data"); return (ARCHIVE_FATAL); } raw->entries_written = 0; a->format_data = raw; a->format_name = "raw"; /* no options exist for this format */ a->format_options = NULL; a->format_write_header = archive_write_raw_header; a->format_write_data = archive_write_raw_data; a->format_finish_entry = NULL; /* nothing needs to be done on closing */ a->format_close = NULL; a->format_free = archive_write_raw_free; a->archive.archive_format = ARCHIVE_FORMAT_RAW; a->archive.archive_format_name = "RAW"; return (ARCHIVE_OK); } static int archive_write_raw_header(struct archive_write *a, struct archive_entry *entry) { struct raw *raw = (struct raw *)a->format_data; if (archive_entry_filetype(entry) != AE_IFREG) { archive_set_error(&a->archive, ERANGE, "Raw format only supports filetype AE_IFREG"); return (ARCHIVE_FATAL); } if (raw->entries_written > 0) { archive_set_error(&a->archive, ERANGE, "Raw format only supports one entry per archive"); return (ARCHIVE_FATAL); } raw->entries_written++; return (ARCHIVE_OK); } static ssize_t archive_write_raw_data(struct archive_write *a, const void *buff, size_t s) { int ret; ret = __archive_write_output(a, buff, s); if (ret >= 0) return (s); else return (ret); } static int archive_write_raw_free(struct archive_write *a) { struct raw *raw; raw = (struct raw *)a->format_data; free(raw); a->format_data = NULL; return (ARCHIVE_OK); }