/*- * Copyright (c) 2003-2007 Tim Kientzle * Copyright (c) 2010 Michihiro NAKAJIMA * 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" __FBSDID("$FreeBSD$"); #include "archive.h" #include "archive_entry.h" #include "archive_private.h" #include "archive_entry_private.h" /* * sparse handling */ void archive_entry_sparse_clear(struct archive_entry *entry) { struct ae_sparse *sp; while (entry->sparse_head != NULL) { sp = entry->sparse_head->next; free(entry->sparse_head); entry->sparse_head = sp; } entry->sparse_tail = &(entry->sparse_head); } void archive_entry_sparse_add_entry(struct archive_entry *entry, int64_t offset, int64_t length) { struct ae_sparse *sp; if ((sp = (struct ae_sparse *)malloc(sizeof(*sp))) == NULL) /* XXX Error XXX */ return; sp->offset = offset; sp->length = length; sp->next = NULL; *entry->sparse_tail = sp; entry->sparse_tail = &(sp->next); } /* * returns number of the sparse entries */ int archive_entry_sparse_count(struct archive_entry *entry) { struct ae_sparse *sp; int count = 0; for (sp = entry->sparse_head; sp != NULL; sp = sp->next) count++; return count; } int archive_entry_sparse_reset(struct archive_entry * entry) { entry->sparse_p = entry->sparse_head; return archive_entry_sparse_count(entry); } int archive_entry_sparse_next(struct archive_entry * entry, int64_t *offset, int64_t *length) { if (entry->sparse_p) { *offset = entry->sparse_p->offset; *length = entry->sparse_p->length; entry->sparse_p = entry->sparse_p->next; return (ARCHIVE_OK); } else { *offset = 0; *length = 0; return (ARCHIVE_WARN); } } /* * end of sparse handling */