Blame libarchive/archive_match.c

Packit Service 1d0348
/*-
Packit Service 1d0348
 * Copyright (c) 2003-2007 Tim Kientzle
Packit Service 1d0348
 * Copyright (c) 2012 Michihiro NAKAJIMA
Packit Service 1d0348
 * All rights reserved.
Packit Service 1d0348
 *
Packit Service 1d0348
 * Redistribution and use in source and binary forms, with or without
Packit Service 1d0348
 * modification, are permitted provided that the following conditions
Packit Service 1d0348
 * are met:
Packit Service 1d0348
 * 1. Redistributions of source code must retain the above copyright
Packit Service 1d0348
 *    notice, this list of conditions and the following disclaimer.
Packit Service 1d0348
 * 2. Redistributions in binary form must reproduce the above copyright
Packit Service 1d0348
 *    notice, this list of conditions and the following disclaimer in the
Packit Service 1d0348
 *    documentation and/or other materials provided with the distribution.
Packit Service 1d0348
 *
Packit Service 1d0348
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
Packit Service 1d0348
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
Packit Service 1d0348
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
Packit Service 1d0348
 * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
Packit Service 1d0348
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
Packit Service 1d0348
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
Packit Service 1d0348
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
Packit Service 1d0348
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit Service 1d0348
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
Packit Service 1d0348
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit Service 1d0348
 */
Packit Service 1d0348
Packit Service 1d0348
#include "archive_platform.h"
Packit Service 1d0348
__FBSDID("$FreeBSD$");
Packit Service 1d0348
Packit Service 1d0348
#ifdef HAVE_ERRNO_H
Packit Service 1d0348
#include <errno.h>
Packit Service 1d0348
#endif
Packit Service 1d0348
#ifdef HAVE_STDLIB_H
Packit Service 1d0348
#include <stdlib.h>
Packit Service 1d0348
#endif
Packit Service 1d0348
#ifdef HAVE_STRING_H
Packit Service 1d0348
#include <string.h>
Packit Service 1d0348
#endif
Packit Service 1d0348
Packit Service 1d0348
#include "archive.h"
Packit Service 1d0348
#include "archive_private.h"
Packit Service 1d0348
#include "archive_entry.h"
Packit Service 1d0348
#include "archive_getdate.h"
Packit Service 1d0348
#include "archive_pathmatch.h"
Packit Service 1d0348
#include "archive_rb.h"
Packit Service 1d0348
#include "archive_string.h"
Packit Service 1d0348
Packit Service 1d0348
struct match {
Packit Service 1d0348
	struct match		*next;
Packit Service 1d0348
	int			 matches;
Packit Service 1d0348
	struct archive_mstring	 pattern;
Packit Service 1d0348
};
Packit Service 1d0348
Packit Service 1d0348
struct match_list {
Packit Service 1d0348
	struct match		*first;
Packit Service 1d0348
	struct match		**last;
Packit Service 1d0348
	int			 count;
Packit Service 1d0348
	int			 unmatched_count;
Packit Service 1d0348
	struct match		*unmatched_next;
Packit Service 1d0348
	int			 unmatched_eof;
Packit Service 1d0348
};
Packit Service 1d0348
Packit Service 1d0348
struct match_file {
Packit Service 1d0348
	struct archive_rb_node	 node;
Packit Service 1d0348
	struct match_file	*next;
Packit Service 1d0348
	struct archive_mstring	 pathname;
Packit Service 1d0348
	int			 flag;
Packit Service 1d0348
	time_t			 mtime_sec;
Packit Service 1d0348
	long			 mtime_nsec;
Packit Service 1d0348
	time_t			 ctime_sec;
Packit Service 1d0348
	long			 ctime_nsec;
Packit Service 1d0348
};
Packit Service 1d0348
Packit Service 1d0348
struct entry_list {
Packit Service 1d0348
	struct match_file	*first;
Packit Service 1d0348
	struct match_file	**last;
Packit Service 1d0348
	int			 count;
Packit Service 1d0348
};
Packit Service 1d0348
Packit Service 1d0348
struct id_array {
Packit Service 1d0348
	size_t			 size;/* Allocated size */
Packit Service 1d0348
	size_t			 count;
Packit Service 1d0348
	int64_t			*ids;
Packit Service 1d0348
};
Packit Service 1d0348
Packit Service 1d0348
#define PATTERN_IS_SET		1
Packit Service 1d0348
#define TIME_IS_SET		2
Packit Service 1d0348
#define ID_IS_SET		4
Packit Service 1d0348
Packit Service 1d0348
struct archive_match {
Packit Service 1d0348
	struct archive		 archive;
Packit Service 1d0348
Packit Service 1d0348
	/* exclusion/inclusion set flag. */
Packit Service 1d0348
	int			 setflag;
Packit Service 1d0348
Packit Service 1d0348
	/*
Packit Service 1d0348
	 * Matching filename patterns.
Packit Service 1d0348
	 */
Packit Service 1d0348
	struct match_list	 exclusions;
Packit Service 1d0348
	struct match_list	 inclusions;
Packit Service 1d0348
Packit Service 1d0348
	/*
Packit Service 1d0348
	 * Matching time stamps.
Packit Service 1d0348
	 */
Packit Service 1d0348
	time_t			 now;
Packit Service 1d0348
	int			 newer_mtime_filter;
Packit Service 1d0348
	time_t			 newer_mtime_sec;
Packit Service 1d0348
	long			 newer_mtime_nsec;
Packit Service 1d0348
	int			 newer_ctime_filter;
Packit Service 1d0348
	time_t			 newer_ctime_sec;
Packit Service 1d0348
	long			 newer_ctime_nsec;
Packit Service 1d0348
	int			 older_mtime_filter;
Packit Service 1d0348
	time_t			 older_mtime_sec;
Packit Service 1d0348
	long			 older_mtime_nsec;
Packit Service 1d0348
	int			 older_ctime_filter;
Packit Service 1d0348
	time_t			 older_ctime_sec;
Packit Service 1d0348
	long			 older_ctime_nsec;
Packit Service 1d0348
	/*
Packit Service 1d0348
	 * Matching time stamps with its filename.
Packit Service 1d0348
	 */
Packit Service 1d0348
	struct archive_rb_tree	 exclusion_tree;
Packit Service 1d0348
	struct entry_list 	 exclusion_entry_list;
Packit Service 1d0348
Packit Service 1d0348
	/*
Packit Service 1d0348
	 * Matching file owners.
Packit Service 1d0348
	 */
Packit Service 1d0348
	struct id_array 	 inclusion_uids;
Packit Service 1d0348
	struct id_array 	 inclusion_gids;
Packit Service 1d0348
	struct match_list	 inclusion_unames;
Packit Service 1d0348
	struct match_list	 inclusion_gnames;
Packit Service 1d0348
};
Packit Service 1d0348
Packit Service 1d0348
static int	add_pattern_from_file(struct archive_match *,
Packit Service 1d0348
		    struct match_list *, int, const void *, int);
Packit Service 1d0348
static int	add_entry(struct archive_match *, int,
Packit Service 1d0348
		    struct archive_entry *);
Packit Service 1d0348
static int	add_owner_id(struct archive_match *, struct id_array *,
Packit Service 1d0348
		    int64_t);
Packit Service 1d0348
static int	add_owner_name(struct archive_match *, struct match_list *,
Packit Service 1d0348
		    int, const void *);
Packit Service 1d0348
static int	add_pattern_mbs(struct archive_match *, struct match_list *,
Packit Service 1d0348
		    const char *);
Packit Service 1d0348
static int	add_pattern_wcs(struct archive_match *, struct match_list *,
Packit Service 1d0348
		    const wchar_t *);
Packit Service 1d0348
static int	cmp_key_mbs(const struct archive_rb_node *, const void *);
Packit Service 1d0348
static int	cmp_key_wcs(const struct archive_rb_node *, const void *);
Packit Service 1d0348
static int	cmp_node_mbs(const struct archive_rb_node *,
Packit Service 1d0348
		    const struct archive_rb_node *);
Packit Service 1d0348
static int	cmp_node_wcs(const struct archive_rb_node *,
Packit Service 1d0348
		    const struct archive_rb_node *);
Packit Service 1d0348
static void	entry_list_add(struct entry_list *, struct match_file *);
Packit Service 1d0348
static void	entry_list_free(struct entry_list *);
Packit Service 1d0348
static void	entry_list_init(struct entry_list *);
Packit Service 1d0348
static int	error_nomem(struct archive_match *);
Packit Service 1d0348
static void	match_list_add(struct match_list *, struct match *);
Packit Service 1d0348
static void	match_list_free(struct match_list *);
Packit Service 1d0348
static void	match_list_init(struct match_list *);
Packit Service 1d0348
static int	match_list_unmatched_inclusions_next(struct archive_match *,
Packit Service 1d0348
		    struct match_list *, int, const void **);
Packit Service 1d0348
static int	match_owner_id(struct id_array *, int64_t);
Packit Service 1d0348
#if !defined(_WIN32) || defined(__CYGWIN__)
Packit Service 1d0348
static int	match_owner_name_mbs(struct archive_match *,
Packit Service 1d0348
		    struct match_list *, const char *);
Packit Service 1d0348
#else
Packit Service 1d0348
static int	match_owner_name_wcs(struct archive_match *,
Packit Service 1d0348
		    struct match_list *, const wchar_t *);
Packit Service 1d0348
#endif
Packit Service 1d0348
static int	match_path_exclusion(struct archive_match *,
Packit Service 1d0348
		    struct match *, int, const void *);
Packit Service 1d0348
static int	match_path_inclusion(struct archive_match *,
Packit Service 1d0348
		    struct match *, int, const void *);
Packit Service 1d0348
static int	owner_excluded(struct archive_match *,
Packit Service 1d0348
		    struct archive_entry *);
Packit Service 1d0348
static int	path_excluded(struct archive_match *, int, const void *);
Packit Service 1d0348
static int	set_timefilter(struct archive_match *, int, time_t, long,
Packit Service 1d0348
		    time_t, long);
Packit Service 1d0348
static int	set_timefilter_pathname_mbs(struct archive_match *,
Packit Service 1d0348
		    int, const char *);
Packit Service 1d0348
static int	set_timefilter_pathname_wcs(struct archive_match *,
Packit Service 1d0348
		    int, const wchar_t *);
Packit Service 1d0348
static int	set_timefilter_date(struct archive_match *, int, const char *);
Packit Service 1d0348
static int	set_timefilter_date_w(struct archive_match *, int,
Packit Service 1d0348
		    const wchar_t *);
Packit Service 1d0348
static int	time_excluded(struct archive_match *,
Packit Service 1d0348
		    struct archive_entry *);
Packit Service 1d0348
static int	validate_time_flag(struct archive *, int, const char *);
Packit Service 1d0348
Packit Service 1d0348
#define get_date __archive_get_date
Packit Service 1d0348
Packit Service 1d0348
static const struct archive_rb_tree_ops rb_ops_mbs = {
Packit Service 1d0348
	cmp_node_mbs, cmp_key_mbs
Packit Service 1d0348
};
Packit Service 1d0348
Packit Service 1d0348
static const struct archive_rb_tree_ops rb_ops_wcs = {
Packit Service 1d0348
	cmp_node_wcs, cmp_key_wcs
Packit Service 1d0348
};
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * The matching logic here needs to be re-thought.  I started out to
Packit Service 1d0348
 * try to mimic gtar's matching logic, but it's not entirely
Packit Service 1d0348
 * consistent.  In particular 'tar -t' and 'tar -x' interpret patterns
Packit Service 1d0348
 * on the command line as anchored, but --exclude doesn't.
Packit Service 1d0348
 */
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
error_nomem(struct archive_match *a)
Packit Service 1d0348
{
Packit Service 1d0348
	archive_set_error(&(a->archive), ENOMEM, "No memory");
Packit Service 1d0348
	a->archive.state = ARCHIVE_STATE_FATAL;
Packit Service 1d0348
	return (ARCHIVE_FATAL);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Create an ARCHIVE_MATCH object.
Packit Service 1d0348
 */
Packit Service 1d0348
struct archive *
Packit Service 1d0348
archive_match_new(void)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	a = (struct archive_match *)calloc(1, sizeof(*a));
Packit Service 1d0348
	if (a == NULL)
Packit Service 1d0348
		return (NULL);
Packit Service 1d0348
	a->archive.magic = ARCHIVE_MATCH_MAGIC;
Packit Service 1d0348
	a->archive.state = ARCHIVE_STATE_NEW;
Packit Service 1d0348
	match_list_init(&(a->inclusions));
Packit Service 1d0348
	match_list_init(&(a->exclusions));
Packit Service 1d0348
	__archive_rb_tree_init(&(a->exclusion_tree), &rb_ops_mbs);
Packit Service 1d0348
	entry_list_init(&(a->exclusion_entry_list));
Packit Service 1d0348
	match_list_init(&(a->inclusion_unames));
Packit Service 1d0348
	match_list_init(&(a->inclusion_gnames));
Packit Service 1d0348
	time(&a->now);
Packit Service 1d0348
	return (&(a->archive));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Free an ARCHIVE_MATCH object.
Packit Service 1d0348
 */
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_free(struct archive *_a)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	if (_a == NULL)
Packit Service 1d0348
		return (ARCHIVE_OK);
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_ANY | ARCHIVE_STATE_FATAL, "archive_match_free");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	match_list_free(&(a->inclusions));
Packit Service 1d0348
	match_list_free(&(a->exclusions));
Packit Service 1d0348
	entry_list_free(&(a->exclusion_entry_list));
Packit Service 1d0348
	free(a->inclusion_uids.ids);
Packit Service 1d0348
	free(a->inclusion_gids.ids);
Packit Service 1d0348
	match_list_free(&(a->inclusion_unames));
Packit Service 1d0348
	match_list_free(&(a->inclusion_gnames));
Packit Service 1d0348
	free(a);
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Convenience function to perform all exclusion tests.
Packit Service 1d0348
 *
Packit Service 1d0348
 * Returns 1 if archive entry is excluded.
Packit Service 1d0348
 * Returns 0 if archive entry is not excluded.
Packit Service 1d0348
 * Returns <0 if something error happened.
Packit Service 1d0348
 */
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_excluded(struct archive *_a, struct archive_entry *entry)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_excluded_ae");
Packit Service 1d0348
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	if (entry == NULL) {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "entry is NULL");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	r = 0;
Packit Service 1d0348
	if (a->setflag & PATTERN_IS_SET) {
Packit Service 1d0348
#if defined(_WIN32) && !defined(__CYGWIN__)
Packit Service 1d0348
		r = path_excluded(a, 0, archive_entry_pathname_w(entry));
Packit Service 1d0348
#else
Packit Service 1d0348
		r = path_excluded(a, 1, archive_entry_pathname(entry));
Packit Service 1d0348
#endif
Packit Service 1d0348
		if (r != 0)
Packit Service 1d0348
			return (r);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	if (a->setflag & TIME_IS_SET) {
Packit Service 1d0348
		r = time_excluded(a, entry);
Packit Service 1d0348
		if (r != 0)
Packit Service 1d0348
			return (r);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	if (a->setflag & ID_IS_SET)
Packit Service 1d0348
		r = owner_excluded(a, entry);
Packit Service 1d0348
	return (r);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Utility functions to manage exclusion/inclusion patterns
Packit Service 1d0348
 */
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_exclude_pattern(struct archive *_a, const char *pattern)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_exclude_pattern");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	if (pattern == NULL || *pattern == '\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pattern is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	if ((r = add_pattern_mbs(a, &(a->exclusions), pattern)) != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_exclude_pattern_w(struct archive *_a, const wchar_t *pattern)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_exclude_pattern_w");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	if (pattern == NULL || *pattern == L'\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pattern is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	if ((r = add_pattern_wcs(a, &(a->exclusions), pattern)) != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_exclude_pattern_from_file(struct archive *_a,
Packit Service 1d0348
    const char *pathname, int nullSeparator)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_exclude_pattern_from_file");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	return add_pattern_from_file(a, &(a->exclusions), 1, pathname,
Packit Service 1d0348
		nullSeparator);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_exclude_pattern_from_file_w(struct archive *_a,
Packit Service 1d0348
    const wchar_t *pathname, int nullSeparator)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_exclude_pattern_from_file_w");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	return add_pattern_from_file(a, &(a->exclusions), 0, pathname,
Packit Service 1d0348
		nullSeparator);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_pattern(struct archive *_a, const char *pattern)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_pattern");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	if (pattern == NULL || *pattern == '\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pattern is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	if ((r = add_pattern_mbs(a, &(a->inclusions), pattern)) != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_pattern_w(struct archive *_a, const wchar_t *pattern)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_pattern_w");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	if (pattern == NULL || *pattern == L'\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pattern is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	if ((r = add_pattern_wcs(a, &(a->inclusions), pattern)) != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_pattern_from_file(struct archive *_a,
Packit Service 1d0348
    const char *pathname, int nullSeparator)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_pattern_from_file");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	return add_pattern_from_file(a, &(a->inclusions), 1, pathname,
Packit Service 1d0348
		nullSeparator);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_pattern_from_file_w(struct archive *_a,
Packit Service 1d0348
    const wchar_t *pathname, int nullSeparator)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_pattern_from_file_w");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	return add_pattern_from_file(a, &(a->inclusions), 0, pathname,
Packit Service 1d0348
		nullSeparator);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Test functions for pathname patterns.
Packit Service 1d0348
 *
Packit Service 1d0348
 * Returns 1 if archive entry is excluded.
Packit Service 1d0348
 * Returns 0 if archive entry is not excluded.
Packit Service 1d0348
 * Returns <0 if something error happened.
Packit Service 1d0348
 */
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_path_excluded(struct archive *_a,
Packit Service 1d0348
    struct archive_entry *entry)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_path_excluded");
Packit Service 1d0348
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	if (entry == NULL) {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "entry is NULL");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* If we don't have exclusion/inclusion pattern set at all,
Packit Service 1d0348
	 * the entry is always not excluded. */
Packit Service 1d0348
	if ((a->setflag & PATTERN_IS_SET) == 0)
Packit Service 1d0348
		return (0);
Packit Service 1d0348
#if defined(_WIN32) && !defined(__CYGWIN__)
Packit Service 1d0348
	return (path_excluded(a, 0, archive_entry_pathname_w(entry)));
Packit Service 1d0348
#else
Packit Service 1d0348
	return (path_excluded(a, 1, archive_entry_pathname(entry)));
Packit Service 1d0348
#endif
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Utility functions to get statistic information for inclusion patterns.
Packit Service 1d0348
 */
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_path_unmatched_inclusions(struct archive *_a)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_unmatched_inclusions");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	return (a->inclusions.unmatched_count);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_path_unmatched_inclusions_next(struct archive *_a,
Packit Service 1d0348
    const char **_p)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
	const void *v;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_unmatched_inclusions_next");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	r = match_list_unmatched_inclusions_next(a, &(a->inclusions), 1, &v);
Packit Service 1d0348
	*_p = (const char *)v;
Packit Service 1d0348
	return (r);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_path_unmatched_inclusions_next_w(struct archive *_a,
Packit Service 1d0348
    const wchar_t **_p)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
	const void *v;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_unmatched_inclusions_next_w");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	r = match_list_unmatched_inclusions_next(a, &(a->inclusions), 0, &v);
Packit Service 1d0348
	*_p = (const wchar_t *)v;
Packit Service 1d0348
	return (r);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Add inclusion/exclusion patterns.
Packit Service 1d0348
 */
Packit Service 1d0348
static int
Packit Service 1d0348
add_pattern_mbs(struct archive_match *a, struct match_list *list,
Packit Service 1d0348
    const char *pattern)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match *match;
Packit Service 1d0348
	size_t len;
Packit Service 1d0348
Packit Service 1d0348
	match = calloc(1, sizeof(*match));
Packit Service 1d0348
	if (match == NULL)
Packit Service 1d0348
		return (error_nomem(a));
Packit Service 1d0348
	/* Both "foo/" and "foo" should match "foo/bar". */
Packit Service 1d0348
	len = strlen(pattern);
Packit Service 1d0348
	if (len && pattern[len - 1] == '/')
Packit Service 1d0348
		--len;
Packit Service 1d0348
	archive_mstring_copy_mbs_len(&(match->pattern), pattern, len);
Packit Service 1d0348
	match_list_add(list, match);
Packit Service 1d0348
	a->setflag |= PATTERN_IS_SET;
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
add_pattern_wcs(struct archive_match *a, struct match_list *list,
Packit Service 1d0348
    const wchar_t *pattern)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match *match;
Packit Service 1d0348
	size_t len;
Packit Service 1d0348
Packit Service 1d0348
	match = calloc(1, sizeof(*match));
Packit Service 1d0348
	if (match == NULL)
Packit Service 1d0348
		return (error_nomem(a));
Packit Service 1d0348
	/* Both "foo/" and "foo" should match "foo/bar". */
Packit Service 1d0348
	len = wcslen(pattern);
Packit Service 1d0348
	if (len && pattern[len - 1] == L'/')
Packit Service 1d0348
		--len;
Packit Service 1d0348
	archive_mstring_copy_wcs_len(&(match->pattern), pattern, len);
Packit Service 1d0348
	match_list_add(list, match);
Packit Service 1d0348
	a->setflag |= PATTERN_IS_SET;
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
add_pattern_from_file(struct archive_match *a, struct match_list *mlist,
Packit Service 1d0348
    int mbs, const void *pathname, int nullSeparator)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive *ar;
Packit Service 1d0348
	struct archive_entry *ae;
Packit Service 1d0348
	struct archive_string as;
Packit Service 1d0348
	const void *buff;
Packit Service 1d0348
	size_t size;
Packit Service 1d0348
	int64_t offset;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	ar = archive_read_new(); 
Packit Service 1d0348
	if (ar == NULL) {
Packit Service 1d0348
		archive_set_error(&(a->archive), ENOMEM, "No memory");
Packit Service 1d0348
		return (ARCHIVE_FATAL);
Packit Service 1d0348
	}
Packit Service 1d0348
	r = archive_read_support_format_raw(ar);
Packit Service 1d0348
	r = archive_read_support_format_empty(ar);
Packit Service 1d0348
	if (r != ARCHIVE_OK) {
Packit Service 1d0348
		archive_copy_error(&(a->archive), ar);
Packit Service 1d0348
		archive_read_free(ar);
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	}
Packit Service 1d0348
	if (mbs)
Packit Service 1d0348
		r = archive_read_open_filename(ar, pathname, 512*20);
Packit Service 1d0348
	else
Packit Service 1d0348
		r = archive_read_open_filename_w(ar, pathname, 512*20);
Packit Service 1d0348
	if (r != ARCHIVE_OK) {
Packit Service 1d0348
		archive_copy_error(&(a->archive), ar);
Packit Service 1d0348
		archive_read_free(ar);
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	}
Packit Service 1d0348
	r = archive_read_next_header(ar, &ae);
Packit Service 1d0348
	if (r != ARCHIVE_OK) {
Packit Service 1d0348
		archive_read_free(ar);
Packit Service 1d0348
		if (r == ARCHIVE_EOF) {
Packit Service 1d0348
			return (ARCHIVE_OK);
Packit Service 1d0348
		} else {
Packit Service 1d0348
			archive_copy_error(&(a->archive), ar);
Packit Service 1d0348
			return (r);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	archive_string_init(&as);
Packit Service 1d0348
Packit Service 1d0348
	while ((r = archive_read_data_block(ar, &buff, &size, &offset))
Packit Service 1d0348
	    == ARCHIVE_OK) {
Packit Service 1d0348
		const char *b = (const char *)buff;
Packit Service 1d0348
Packit Service 1d0348
		while (size) {
Packit Service 1d0348
			const char *s = (const char *)b;
Packit Service 1d0348
			size_t length = 0;
Packit Service 1d0348
			int found_separator = 0;
Packit Service 1d0348
Packit Service 1d0348
			while (length < size) {
Packit Service 1d0348
				if (nullSeparator) {
Packit Service 1d0348
					if (*b == '\0') {
Packit Service 1d0348
						found_separator = 1;
Packit Service 1d0348
						break;
Packit Service 1d0348
					}
Packit Service 1d0348
				} else {
Packit Service 1d0348
			            	if (*b == 0x0d || *b == 0x0a) {
Packit Service 1d0348
						found_separator = 1;
Packit Service 1d0348
						break;
Packit Service 1d0348
					}
Packit Service 1d0348
				}
Packit Service 1d0348
				b++;
Packit Service 1d0348
				length++;
Packit Service 1d0348
			}
Packit Service 1d0348
			if (!found_separator) {
Packit Service 1d0348
				archive_strncat(&as, s, length);
Packit Service 1d0348
				/* Read next data block. */
Packit Service 1d0348
				break;
Packit Service 1d0348
			}
Packit Service 1d0348
			b++;
Packit Service 1d0348
			size -= length + 1;
Packit Service 1d0348
			archive_strncat(&as, s, length);
Packit Service 1d0348
Packit Service 1d0348
			/* If the line is not empty, add the pattern. */
Packit Service 1d0348
			if (archive_strlen(&as) > 0) {
Packit Service 1d0348
				/* Add pattern. */
Packit Service 1d0348
				r = add_pattern_mbs(a, mlist, as.s);
Packit Service 1d0348
				if (r != ARCHIVE_OK) {
Packit Service 1d0348
					archive_read_free(ar);
Packit Service 1d0348
					archive_string_free(&as);
Packit Service 1d0348
					return (r);
Packit Service 1d0348
				}
Packit Service 1d0348
				archive_string_empty(&as);
Packit Service 1d0348
			}
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* If an error occurred, report it immediately. */
Packit Service 1d0348
	if (r < ARCHIVE_OK) {
Packit Service 1d0348
		archive_copy_error(&(a->archive), ar);
Packit Service 1d0348
		archive_read_free(ar);
Packit Service 1d0348
		archive_string_free(&as);
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* If the line is not empty, add the pattern. */
Packit Service 1d0348
	if (r == ARCHIVE_EOF && archive_strlen(&as) > 0) {
Packit Service 1d0348
		/* Add pattern. */
Packit Service 1d0348
		r = add_pattern_mbs(a, mlist, as.s);
Packit Service 1d0348
		if (r != ARCHIVE_OK) {
Packit Service 1d0348
			archive_read_free(ar);
Packit Service 1d0348
			archive_string_free(&as);
Packit Service 1d0348
			return (r);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	archive_read_free(ar);
Packit Service 1d0348
	archive_string_free(&as);
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Test if pathname is excluded by inclusion/exclusion patterns.
Packit Service 1d0348
 */
Packit Service 1d0348
static int
Packit Service 1d0348
path_excluded(struct archive_match *a, int mbs, const void *pathname)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match *match;
Packit Service 1d0348
	struct match *matched;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	if (a == NULL)
Packit Service 1d0348
		return (0);
Packit Service 1d0348
Packit Service 1d0348
	/* Mark off any unmatched inclusions. */
Packit Service 1d0348
	/* In particular, if a filename does appear in the archive and
Packit Service 1d0348
	 * is explicitly included and excluded, then we don't report
Packit Service 1d0348
	 * it as missing even though we don't extract it.
Packit Service 1d0348
	 */
Packit Service 1d0348
	matched = NULL;
Packit Service 1d0348
	for (match = a->inclusions.first; match != NULL;
Packit Service 1d0348
	    match = match->next){
Packit Service 1d0348
		if (match->matches == 0 &&
Packit Service 1d0348
		    (r = match_path_inclusion(a, match, mbs, pathname)) != 0) {
Packit Service 1d0348
			if (r < 0)
Packit Service 1d0348
				return (r);
Packit Service 1d0348
			a->inclusions.unmatched_count--;
Packit Service 1d0348
			match->matches++;
Packit Service 1d0348
			matched = match;
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* Exclusions take priority */
Packit Service 1d0348
	for (match = a->exclusions.first; match != NULL;
Packit Service 1d0348
	    match = match->next){
Packit Service 1d0348
		r = match_path_exclusion(a, match, mbs, pathname);
Packit Service 1d0348
		if (r)
Packit Service 1d0348
			return (r);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* It's not excluded and we found an inclusion above, so it's
Packit Service 1d0348
	 * included. */
Packit Service 1d0348
	if (matched != NULL)
Packit Service 1d0348
		return (0);
Packit Service 1d0348
Packit Service 1d0348
Packit Service 1d0348
	/* We didn't find an unmatched inclusion, check the remaining ones. */
Packit Service 1d0348
	for (match = a->inclusions.first; match != NULL;
Packit Service 1d0348
	    match = match->next){
Packit Service 1d0348
		/* We looked at previously-unmatched inclusions already. */
Packit Service 1d0348
		if (match->matches > 0 &&
Packit Service 1d0348
		    (r = match_path_inclusion(a, match, mbs, pathname)) != 0) {
Packit Service 1d0348
			if (r < 0)
Packit Service 1d0348
				return (r);
Packit Service 1d0348
			match->matches++;
Packit Service 1d0348
			return (0);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* If there were inclusions, default is to exclude. */
Packit Service 1d0348
	if (a->inclusions.first != NULL)
Packit Service 1d0348
	    return (1);
Packit Service 1d0348
Packit Service 1d0348
	/* No explicit inclusions, default is to match. */
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * This is a little odd, but it matches the default behavior of
Packit Service 1d0348
 * gtar.  In particular, 'a*b' will match 'foo/a1111/222b/bar'
Packit Service 1d0348
 *
Packit Service 1d0348
 */
Packit Service 1d0348
static int
Packit Service 1d0348
match_path_exclusion(struct archive_match *a, struct match *m,
Packit Service 1d0348
    int mbs, const void *pn)
Packit Service 1d0348
{
Packit Service 1d0348
	int flag = PATHMATCH_NO_ANCHOR_START | PATHMATCH_NO_ANCHOR_END;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	if (mbs) {
Packit Service 1d0348
		const char *p;
Packit Service 1d0348
		r = archive_mstring_get_mbs(&(a->archive), &(m->pattern), &p);
Packit Service 1d0348
		if (r == 0)
Packit Service 1d0348
			return (archive_pathmatch(p, (const char *)pn, flag));
Packit Service 1d0348
	} else {
Packit Service 1d0348
		const wchar_t *p;
Packit Service 1d0348
		r = archive_mstring_get_wcs(&(a->archive), &(m->pattern), &p);
Packit Service 1d0348
		if (r == 0)
Packit Service 1d0348
			return (archive_pathmatch_w(p, (const wchar_t *)pn,
Packit Service 1d0348
				flag));
Packit Service 1d0348
	}
Packit Service 1d0348
	if (errno == ENOMEM)
Packit Service 1d0348
		return (error_nomem(a));
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Again, mimic gtar:  inclusions are always anchored (have to match
Packit Service 1d0348
 * the beginning of the path) even though exclusions are not anchored.
Packit Service 1d0348
 */
Packit Service 1d0348
static int
Packit Service 1d0348
match_path_inclusion(struct archive_match *a, struct match *m,
Packit Service 1d0348
    int mbs, const void *pn)
Packit Service 1d0348
{
Packit Service 1d0348
	int flag = PATHMATCH_NO_ANCHOR_END;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	if (mbs) {
Packit Service 1d0348
		const char *p;
Packit Service 1d0348
		r = archive_mstring_get_mbs(&(a->archive), &(m->pattern), &p);
Packit Service 1d0348
		if (r == 0)
Packit Service 1d0348
			return (archive_pathmatch(p, (const char *)pn, flag));
Packit Service 1d0348
	} else {
Packit Service 1d0348
		const wchar_t *p;
Packit Service 1d0348
		r = archive_mstring_get_wcs(&(a->archive), &(m->pattern), &p);
Packit Service 1d0348
		if (r == 0)
Packit Service 1d0348
			return (archive_pathmatch_w(p, (const wchar_t *)pn,
Packit Service 1d0348
				flag));
Packit Service 1d0348
	}
Packit Service 1d0348
	if (errno == ENOMEM)
Packit Service 1d0348
		return (error_nomem(a));
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static void
Packit Service 1d0348
match_list_init(struct match_list *list)
Packit Service 1d0348
{
Packit Service 1d0348
	list->first = NULL;
Packit Service 1d0348
	list->last = &(list->first);
Packit Service 1d0348
	list->count = 0;
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static void
Packit Service 1d0348
match_list_free(struct match_list *list)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match *p, *q;
Packit Service 1d0348
Packit Service 1d0348
	for (p = list->first; p != NULL; ) {
Packit Service 1d0348
		q = p;
Packit Service 1d0348
		p = p->next;
Packit Service 1d0348
		archive_mstring_clean(&(q->pattern));
Packit Service 1d0348
		free(q);
Packit Service 1d0348
	}
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static void
Packit Service 1d0348
match_list_add(struct match_list *list, struct match *m)
Packit Service 1d0348
{
Packit Service 1d0348
	*list->last = m;
Packit Service 1d0348
	list->last = &(m->next);
Packit Service 1d0348
	list->count++;
Packit Service 1d0348
	list->unmatched_count++;
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
match_list_unmatched_inclusions_next(struct archive_match *a,
Packit Service 1d0348
    struct match_list *list, int mbs, const void **vp)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match *m;
Packit Service 1d0348
Packit Service 1d0348
	*vp = NULL;
Packit Service 1d0348
	if (list->unmatched_eof) {
Packit Service 1d0348
		list->unmatched_eof = 0;
Packit Service 1d0348
		return (ARCHIVE_EOF);
Packit Service 1d0348
	}
Packit Service 1d0348
	if (list->unmatched_next == NULL) {
Packit Service 1d0348
		if (list->unmatched_count == 0)
Packit Service 1d0348
			return (ARCHIVE_EOF);
Packit Service 1d0348
		list->unmatched_next = list->first;
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	for (m = list->unmatched_next; m != NULL; m = m->next) {
Packit Service 1d0348
		int r;
Packit Service 1d0348
Packit Service 1d0348
		if (m->matches)
Packit Service 1d0348
			continue;
Packit Service 1d0348
		if (mbs) {
Packit Service 1d0348
			const char *p;
Packit Service 1d0348
			r = archive_mstring_get_mbs(&(a->archive),
Packit Service 1d0348
				&(m->pattern), &p);
Packit Service 1d0348
			if (r < 0 && errno == ENOMEM)
Packit Service 1d0348
				return (error_nomem(a));
Packit Service 1d0348
			if (p == NULL)
Packit Service 1d0348
				p = "";
Packit Service 1d0348
			*vp = p;
Packit Service 1d0348
		} else {
Packit Service 1d0348
			const wchar_t *p;
Packit Service 1d0348
			r = archive_mstring_get_wcs(&(a->archive),
Packit Service 1d0348
				&(m->pattern), &p);
Packit Service 1d0348
			if (r < 0 && errno == ENOMEM)
Packit Service 1d0348
				return (error_nomem(a));
Packit Service 1d0348
			if (p == NULL)
Packit Service 1d0348
				p = L"";
Packit Service 1d0348
			*vp = p;
Packit Service 1d0348
		}
Packit Service 1d0348
		list->unmatched_next = m->next;
Packit Service 1d0348
		if (list->unmatched_next == NULL)
Packit Service 1d0348
			/* To return EOF next time. */
Packit Service 1d0348
			list->unmatched_eof = 1;
Packit Service 1d0348
		return (ARCHIVE_OK);
Packit Service 1d0348
	}
Packit Service 1d0348
	list->unmatched_next = NULL;
Packit Service 1d0348
	return (ARCHIVE_EOF);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Utility functions to manage inclusion timestamps.
Packit Service 1d0348
 */
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_time(struct archive *_a, int flag, time_t sec,
Packit Service 1d0348
    long nsec)
Packit Service 1d0348
{
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	r = validate_time_flag(_a, flag, "archive_match_include_time");
Packit Service 1d0348
	if (r != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return set_timefilter((struct archive_match *)_a, flag,
Packit Service 1d0348
			sec, nsec, sec, nsec);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_date(struct archive *_a, int flag,
Packit Service 1d0348
    const char *datestr)
Packit Service 1d0348
{
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	r = validate_time_flag(_a, flag, "archive_match_include_date");
Packit Service 1d0348
	if (r != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return set_timefilter_date((struct archive_match *)_a, flag, datestr);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_date_w(struct archive *_a, int flag,
Packit Service 1d0348
    const wchar_t *datestr)
Packit Service 1d0348
{
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	r = validate_time_flag(_a, flag, "archive_match_include_date_w");
Packit Service 1d0348
	if (r != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
Packit Service 1d0348
	return set_timefilter_date_w((struct archive_match *)_a, flag, datestr);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_file_time(struct archive *_a, int flag,
Packit Service 1d0348
    const char *pathname)
Packit Service 1d0348
{
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	r = validate_time_flag(_a, flag, "archive_match_include_file_time");
Packit Service 1d0348
	if (r != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return set_timefilter_pathname_mbs((struct archive_match *)_a,
Packit Service 1d0348
			flag, pathname);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_file_time_w(struct archive *_a, int flag,
Packit Service 1d0348
    const wchar_t *pathname)
Packit Service 1d0348
{
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	r = validate_time_flag(_a, flag, "archive_match_include_file_time_w");
Packit Service 1d0348
	if (r != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return set_timefilter_pathname_wcs((struct archive_match *)_a,
Packit Service 1d0348
			flag, pathname);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_exclude_entry(struct archive *_a, int flag,
Packit Service 1d0348
    struct archive_entry *entry)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_time_include_entry");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
Packit Service 1d0348
	if (entry == NULL) {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "entry is NULL");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	r = validate_time_flag(_a, flag, "archive_match_exclude_entry");
Packit Service 1d0348
	if (r != ARCHIVE_OK)
Packit Service 1d0348
		return (r);
Packit Service 1d0348
	return (add_entry(a, flag, entry));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Test function for time stamps.
Packit Service 1d0348
 *
Packit Service 1d0348
 * Returns 1 if archive entry is excluded.
Packit Service 1d0348
 * Returns 0 if archive entry is not excluded.
Packit Service 1d0348
 * Returns <0 if something error happened.
Packit Service 1d0348
 */
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_time_excluded(struct archive *_a,
Packit Service 1d0348
    struct archive_entry *entry)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_time_excluded_ae");
Packit Service 1d0348
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	if (entry == NULL) {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "entry is NULL");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* If we don't have inclusion time set at all, the entry is always
Packit Service 1d0348
	 * not excluded. */
Packit Service 1d0348
	if ((a->setflag & TIME_IS_SET) == 0)
Packit Service 1d0348
		return (0);
Packit Service 1d0348
	return (time_excluded(a, entry));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
validate_time_flag(struct archive *_a, int flag, const char *_fn)
Packit Service 1d0348
{
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, _fn);
Packit Service 1d0348
Packit Service 1d0348
	/* Check a type of time. */
Packit Service 1d0348
	if (flag &
Packit Service 1d0348
	   ((~(ARCHIVE_MATCH_MTIME | ARCHIVE_MATCH_CTIME)) & 0xff00)) {
Packit Service 1d0348
		archive_set_error(_a, EINVAL, "Invalid time flag");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	if ((flag & (ARCHIVE_MATCH_MTIME | ARCHIVE_MATCH_CTIME)) == 0) {
Packit Service 1d0348
		archive_set_error(_a, EINVAL, "No time flag");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* Check a type of comparison. */
Packit Service 1d0348
	if (flag &
Packit Service 1d0348
	   ((~(ARCHIVE_MATCH_NEWER | ARCHIVE_MATCH_OLDER
Packit Service 1d0348
			| ARCHIVE_MATCH_EQUAL)) & 0x00ff)) {
Packit Service 1d0348
		archive_set_error(_a, EINVAL, "Invalid comparison flag");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	if ((flag & (ARCHIVE_MATCH_NEWER | ARCHIVE_MATCH_OLDER
Packit Service 1d0348
	    | ARCHIVE_MATCH_EQUAL)) == 0) {
Packit Service 1d0348
		archive_set_error(_a, EINVAL, "No comparison flag");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
#define JUST_EQUAL(t) (((t) &  (ARCHIVE_MATCH_EQUAL |\
Packit Service 1d0348
	ARCHIVE_MATCH_NEWER | ARCHIVE_MATCH_OLDER)) == ARCHIVE_MATCH_EQUAL)
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter(struct archive_match *a, int timetype,
Packit Service 1d0348
    time_t mtime_sec, long mtime_nsec, time_t ctime_sec, long ctime_nsec)
Packit Service 1d0348
{
Packit Service 1d0348
	if (timetype & ARCHIVE_MATCH_MTIME) {
Packit Service 1d0348
		if ((timetype & ARCHIVE_MATCH_NEWER) || JUST_EQUAL(timetype)) {
Packit Service 1d0348
			a->newer_mtime_filter = timetype;
Packit Service 1d0348
			a->newer_mtime_sec = mtime_sec;
Packit Service 1d0348
			a->newer_mtime_nsec = mtime_nsec;
Packit Service 1d0348
			a->setflag |= TIME_IS_SET;
Packit Service 1d0348
		}
Packit Service 1d0348
		if ((timetype & ARCHIVE_MATCH_OLDER) || JUST_EQUAL(timetype)) {
Packit Service 1d0348
			a->older_mtime_filter = timetype;
Packit Service 1d0348
			a->older_mtime_sec = mtime_sec;
Packit Service 1d0348
			a->older_mtime_nsec = mtime_nsec;
Packit Service 1d0348
			a->setflag |= TIME_IS_SET;
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	if (timetype & ARCHIVE_MATCH_CTIME) {
Packit Service 1d0348
		if ((timetype & ARCHIVE_MATCH_NEWER) || JUST_EQUAL(timetype)) {
Packit Service 1d0348
			a->newer_ctime_filter = timetype;
Packit Service 1d0348
			a->newer_ctime_sec = ctime_sec;
Packit Service 1d0348
			a->newer_ctime_nsec = ctime_nsec;
Packit Service 1d0348
			a->setflag |= TIME_IS_SET;
Packit Service 1d0348
		}
Packit Service 1d0348
		if ((timetype & ARCHIVE_MATCH_OLDER) || JUST_EQUAL(timetype)) {
Packit Service 1d0348
			a->older_ctime_filter = timetype;
Packit Service 1d0348
			a->older_ctime_sec = ctime_sec;
Packit Service 1d0348
			a->older_ctime_nsec = ctime_nsec;
Packit Service 1d0348
			a->setflag |= TIME_IS_SET;
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter_date(struct archive_match *a, int timetype, const char *datestr)
Packit Service 1d0348
{
Packit Service 1d0348
	time_t t;
Packit Service 1d0348
Packit Service 1d0348
	if (datestr == NULL || *datestr == '\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "date is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	t = get_date(a->now, datestr);
Packit Service 1d0348
	if (t == (time_t)-1) {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "invalid date string");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	return set_timefilter(a, timetype, t, 0, t, 0);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter_date_w(struct archive_match *a, int timetype,
Packit Service 1d0348
    const wchar_t *datestr)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_string as;
Packit Service 1d0348
	time_t t;
Packit Service 1d0348
Packit Service 1d0348
	if (datestr == NULL || *datestr == L'\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "date is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	archive_string_init(&as);
Packit Service 1d0348
	if (archive_string_append_from_wcs(&as, datestr, wcslen(datestr)) < 0) {
Packit Service 1d0348
		archive_string_free(&as);
Packit Service 1d0348
		if (errno == ENOMEM)
Packit Service 1d0348
			return (error_nomem(a));
Packit Service 1d0348
		archive_set_error(&(a->archive), -1,
Packit Service 1d0348
		    "Failed to convert WCS to MBS");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	t = get_date(a->now, as.s);
Packit Service 1d0348
	archive_string_free(&as);
Packit Service 1d0348
	if (t == (time_t)-1) {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "invalid date string");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	return set_timefilter(a, timetype, t, 0, t, 0);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
#if defined(_WIN32) && !defined(__CYGWIN__)
Packit Service 1d0348
#define EPOC_TIME ARCHIVE_LITERAL_ULL(116444736000000000)
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter_find_data(struct archive_match *a, int timetype,
Packit Service 1d0348
    DWORD ftLastWriteTime_dwHighDateTime, DWORD ftLastWriteTime_dwLowDateTime,
Packit Service 1d0348
    DWORD ftCreationTime_dwHighDateTime, DWORD ftCreationTime_dwLowDateTime)
Packit Service 1d0348
{
Packit Service 1d0348
	ULARGE_INTEGER utc;
Packit Service 1d0348
	time_t ctime_sec, mtime_sec;
Packit Service 1d0348
	long ctime_ns, mtime_ns;
Packit Service 1d0348
Packit Service 1d0348
	utc.HighPart = ftCreationTime_dwHighDateTime;
Packit Service 1d0348
	utc.LowPart = ftCreationTime_dwLowDateTime;
Packit Service 1d0348
	if (utc.QuadPart >= EPOC_TIME) {
Packit Service 1d0348
		utc.QuadPart -= EPOC_TIME;
Packit Service 1d0348
		ctime_sec = (time_t)(utc.QuadPart / 10000000);
Packit Service 1d0348
		ctime_ns = (long)(utc.QuadPart % 10000000) * 100;
Packit Service 1d0348
	} else {
Packit Service 1d0348
		ctime_sec = 0;
Packit Service 1d0348
		ctime_ns = 0;
Packit Service 1d0348
	}
Packit Service 1d0348
	utc.HighPart = ftLastWriteTime_dwHighDateTime;
Packit Service 1d0348
	utc.LowPart = ftLastWriteTime_dwLowDateTime;
Packit Service 1d0348
	if (utc.QuadPart >= EPOC_TIME) {
Packit Service 1d0348
		utc.QuadPart -= EPOC_TIME;
Packit Service 1d0348
		mtime_sec = (time_t)(utc.QuadPart / 10000000);
Packit Service 1d0348
		mtime_ns = (long)(utc.QuadPart % 10000000) * 100;
Packit Service 1d0348
	} else {
Packit Service 1d0348
		mtime_sec = 0;
Packit Service 1d0348
		mtime_ns = 0;
Packit Service 1d0348
	}
Packit Service 1d0348
	return set_timefilter(a, timetype,
Packit Service 1d0348
			mtime_sec, mtime_ns, ctime_sec, ctime_ns);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter_pathname_mbs(struct archive_match *a, int timetype,
Packit Service 1d0348
    const char *path)
Packit Service 1d0348
{
Packit Service 1d0348
	/* NOTE: stat() on Windows cannot handle nano seconds. */
Packit Service 1d0348
	HANDLE h;
Packit Service 1d0348
	WIN32_FIND_DATAA d;
Packit Service 1d0348
Packit Service 1d0348
	if (path == NULL || *path == '\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pathname is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	h = FindFirstFileA(path, &d);
Packit Service 1d0348
	if (h == INVALID_HANDLE_VALUE) {
Packit Service 1d0348
		la_dosmaperr(GetLastError());
Packit Service 1d0348
		archive_set_error(&(a->archive), errno,
Packit Service 1d0348
		    "Failed to FindFirstFileA");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	FindClose(h);
Packit Service 1d0348
	return set_timefilter_find_data(a, timetype,
Packit Service 1d0348
	    d.ftLastWriteTime.dwHighDateTime, d.ftLastWriteTime.dwLowDateTime,
Packit Service 1d0348
	    d.ftCreationTime.dwHighDateTime, d.ftCreationTime.dwLowDateTime);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter_pathname_wcs(struct archive_match *a, int timetype,
Packit Service 1d0348
    const wchar_t *path)
Packit Service 1d0348
{
Packit Service 1d0348
	HANDLE h;
Packit Service 1d0348
	WIN32_FIND_DATAW d;
Packit Service 1d0348
Packit Service 1d0348
	if (path == NULL || *path == L'\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pathname is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	h = FindFirstFileW(path, &d);
Packit Service 1d0348
	if (h == INVALID_HANDLE_VALUE) {
Packit Service 1d0348
		la_dosmaperr(GetLastError());
Packit Service 1d0348
		archive_set_error(&(a->archive), errno,
Packit Service 1d0348
		    "Failed to FindFirstFile");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	FindClose(h);
Packit Service 1d0348
	return set_timefilter_find_data(a, timetype,
Packit Service 1d0348
	    d.ftLastWriteTime.dwHighDateTime, d.ftLastWriteTime.dwLowDateTime,
Packit Service 1d0348
	    d.ftCreationTime.dwHighDateTime, d.ftCreationTime.dwLowDateTime);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
#else /* _WIN32 && !__CYGWIN__ */
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter_stat(struct archive_match *a, int timetype, struct stat *st)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_entry *ae;
Packit Service 1d0348
	time_t ctime_sec, mtime_sec;
Packit Service 1d0348
	long ctime_ns, mtime_ns;
Packit Service 1d0348
Packit Service 1d0348
	ae = archive_entry_new();
Packit Service 1d0348
	if (ae == NULL)
Packit Service 1d0348
		return (error_nomem(a));
Packit Service 1d0348
	archive_entry_copy_stat(ae, st);
Packit Service 1d0348
	ctime_sec = archive_entry_ctime(ae);
Packit Service 1d0348
	ctime_ns = archive_entry_ctime_nsec(ae);
Packit Service 1d0348
	mtime_sec = archive_entry_mtime(ae);
Packit Service 1d0348
	mtime_ns = archive_entry_mtime_nsec(ae);
Packit Service 1d0348
	archive_entry_free(ae);
Packit Service 1d0348
	return set_timefilter(a, timetype, mtime_sec, mtime_ns,
Packit Service 1d0348
			ctime_sec, ctime_ns);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter_pathname_mbs(struct archive_match *a, int timetype,
Packit Service 1d0348
    const char *path)
Packit Service 1d0348
{
Packit Service 1d0348
	struct stat st;
Packit Service 1d0348
Packit Service 1d0348
	if (path == NULL || *path == '\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pathname is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	if (stat(path, &st) != 0) {
Packit Service 1d0348
		archive_set_error(&(a->archive), errno, "Failed to stat()");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	return (set_timefilter_stat(a, timetype, &st);;
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
set_timefilter_pathname_wcs(struct archive_match *a, int timetype,
Packit Service 1d0348
    const wchar_t *path)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_string as;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	if (path == NULL || *path == L'\0') {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pathname is empty");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* Convert WCS filename to MBS filename. */
Packit Service 1d0348
	archive_string_init(&as);
Packit Service 1d0348
	if (archive_string_append_from_wcs(&as, path, wcslen(path)) < 0) {
Packit Service 1d0348
		archive_string_free(&as);
Packit Service 1d0348
		if (errno == ENOMEM)
Packit Service 1d0348
			return (error_nomem(a));
Packit Service 1d0348
		archive_set_error(&(a->archive), -1,
Packit Service 1d0348
		    "Failed to convert WCS to MBS");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	r = set_timefilter_pathname_mbs(a, timetype, as.s);
Packit Service 1d0348
	archive_string_free(&as);
Packit Service 1d0348
Packit Service 1d0348
	return (r);
Packit Service 1d0348
}
Packit Service 1d0348
#endif /* _WIN32 && !__CYGWIN__ */
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Call back functions for archive_rb.
Packit Service 1d0348
 */
Packit Service 1d0348
static int
Packit Service 1d0348
cmp_node_mbs(const struct archive_rb_node *n1,
Packit Service 1d0348
    const struct archive_rb_node *n2)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match_file *f1 = (struct match_file *)(uintptr_t)n1;
Packit Service 1d0348
	struct match_file *f2 = (struct match_file *)(uintptr_t)n2;
Packit Service 1d0348
	const char *p1, *p2;
Packit Service 1d0348
Packit Service 1d0348
	archive_mstring_get_mbs(NULL, &(f1->pathname), &p1;;
Packit Service 1d0348
	archive_mstring_get_mbs(NULL, &(f2->pathname), &p2;;
Packit Service 1d0348
	if (p1 == NULL)
Packit Service 1d0348
		return (1);
Packit Service 1d0348
	if (p2 == NULL)
Packit Service 1d0348
		return (-1);
Packit Service 1d0348
	return (strcmp(p1, p2));
Packit Service 1d0348
}
Packit Service 1d0348
        
Packit Service 1d0348
static int
Packit Service 1d0348
cmp_key_mbs(const struct archive_rb_node *n, const void *key)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match_file *f = (struct match_file *)(uintptr_t)n;
Packit Service 1d0348
	const char *p;
Packit Service 1d0348
Packit Service 1d0348
	archive_mstring_get_mbs(NULL, &(f->pathname), &p);
Packit Service 1d0348
	if (p == NULL)
Packit Service 1d0348
		return (-1);
Packit Service 1d0348
	return (strcmp(p, (const char *)key));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
cmp_node_wcs(const struct archive_rb_node *n1,
Packit Service 1d0348
    const struct archive_rb_node *n2)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match_file *f1 = (struct match_file *)(uintptr_t)n1;
Packit Service 1d0348
	struct match_file *f2 = (struct match_file *)(uintptr_t)n2;
Packit Service 1d0348
	const wchar_t *p1, *p2;
Packit Service 1d0348
Packit Service 1d0348
	archive_mstring_get_wcs(NULL, &(f1->pathname), &p1;;
Packit Service 1d0348
	archive_mstring_get_wcs(NULL, &(f2->pathname), &p2;;
Packit Service 1d0348
	if (p1 == NULL)
Packit Service 1d0348
		return (1);
Packit Service 1d0348
	if (p2 == NULL)
Packit Service 1d0348
		return (-1);
Packit Service 1d0348
	return (wcscmp(p1, p2));
Packit Service 1d0348
}
Packit Service 1d0348
        
Packit Service 1d0348
static int
Packit Service 1d0348
cmp_key_wcs(const struct archive_rb_node *n, const void *key)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match_file *f = (struct match_file *)(uintptr_t)n;
Packit Service 1d0348
	const wchar_t *p;
Packit Service 1d0348
Packit Service 1d0348
	archive_mstring_get_wcs(NULL, &(f->pathname), &p);
Packit Service 1d0348
	if (p == NULL)
Packit Service 1d0348
		return (-1);
Packit Service 1d0348
	return (wcscmp(p, (const wchar_t *)key));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static void
Packit Service 1d0348
entry_list_init(struct entry_list *list)
Packit Service 1d0348
{
Packit Service 1d0348
	list->first = NULL;
Packit Service 1d0348
	list->last = &(list->first);
Packit Service 1d0348
	list->count = 0;
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static void
Packit Service 1d0348
entry_list_free(struct entry_list *list)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match_file *p, *q;
Packit Service 1d0348
Packit Service 1d0348
	for (p = list->first; p != NULL; ) {
Packit Service 1d0348
		q = p;
Packit Service 1d0348
		p = p->next;
Packit Service 1d0348
		archive_mstring_clean(&(q->pathname));
Packit Service 1d0348
		free(q);
Packit Service 1d0348
	}
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static void
Packit Service 1d0348
entry_list_add(struct entry_list *list, struct match_file *file)
Packit Service 1d0348
{
Packit Service 1d0348
	*list->last = file;
Packit Service 1d0348
	list->last = &(file->next);
Packit Service 1d0348
	list->count++;
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
add_entry(struct archive_match *a, int flag,
Packit Service 1d0348
    struct archive_entry *entry)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match_file *f;
Packit Service 1d0348
	const void *pathname;
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	f = calloc(1, sizeof(*f));
Packit Service 1d0348
	if (f == NULL)
Packit Service 1d0348
		return (error_nomem(a));
Packit Service 1d0348
Packit Service 1d0348
#if defined(_WIN32) && !defined(__CYGWIN__)
Packit Service 1d0348
	pathname = archive_entry_pathname_w(entry);
Packit Service 1d0348
	if (pathname == NULL) {
Packit Service 1d0348
		free(f);
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pathname is NULL");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	archive_mstring_copy_wcs(&(f->pathname), pathname);
Packit Service 1d0348
	a->exclusion_tree.rbt_ops = &rb_ops_wcs;
Packit Service 1d0348
#else
Packit Service 1d0348
	(void)rb_ops_wcs;
Packit Service 1d0348
	pathname = archive_entry_pathname(entry);
Packit Service 1d0348
	if (pathname == NULL) {
Packit Service 1d0348
		free(f);
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "pathname is NULL");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
	archive_mstring_copy_mbs(&(f->pathname), pathname);
Packit Service 1d0348
	a->exclusion_tree.rbt_ops = &rb_ops_mbs;
Packit Service 1d0348
#endif
Packit Service 1d0348
	f->flag = flag;
Packit Service 1d0348
	f->mtime_sec = archive_entry_mtime(entry);
Packit Service 1d0348
	f->mtime_nsec = archive_entry_mtime_nsec(entry);
Packit Service 1d0348
	f->ctime_sec = archive_entry_ctime(entry);
Packit Service 1d0348
	f->ctime_nsec = archive_entry_ctime_nsec(entry);
Packit Service 1d0348
	r = __archive_rb_tree_insert_node(&(a->exclusion_tree), &(f->node));
Packit Service 1d0348
	if (!r) {
Packit Service 1d0348
		struct match_file *f2;
Packit Service 1d0348
Packit Service 1d0348
		/* Get the duplicated file. */
Packit Service 1d0348
		f2 = (struct match_file *)__archive_rb_tree_find_node(
Packit Service 1d0348
			&(a->exclusion_tree), pathname);
Packit Service 1d0348
Packit Service 1d0348
		/*
Packit Service 1d0348
		 * We always overwrite comparison condition.
Packit Service 1d0348
		 * If you do not want to overwrite it, you should not
Packit Service 1d0348
		 * call archive_match_exclude_entry(). We cannot know
Packit Service 1d0348
		 * what behavior you really expect since overwriting
Packit Service 1d0348
		 * condition might be different with the flag.
Packit Service 1d0348
		 */
Packit Service 1d0348
		if (f2 != NULL) {
Packit Service 1d0348
			f2->flag = f->flag;
Packit Service 1d0348
			f2->mtime_sec = f->mtime_sec;
Packit Service 1d0348
			f2->mtime_nsec = f->mtime_nsec;
Packit Service 1d0348
			f2->ctime_sec = f->ctime_sec;
Packit Service 1d0348
			f2->ctime_nsec = f->ctime_nsec;
Packit Service 1d0348
		}
Packit Service 1d0348
		/* Release the duplicated file. */
Packit Service 1d0348
		archive_mstring_clean(&(f->pathname));
Packit Service 1d0348
		free(f);
Packit Service 1d0348
		return (ARCHIVE_OK);
Packit Service 1d0348
	}
Packit Service 1d0348
	entry_list_add(&(a->exclusion_entry_list), f);
Packit Service 1d0348
	a->setflag |= TIME_IS_SET;
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Test if entry is excluded by its timestamp.
Packit Service 1d0348
 */
Packit Service 1d0348
static int
Packit Service 1d0348
time_excluded(struct archive_match *a, struct archive_entry *entry)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match_file *f;
Packit Service 1d0348
	const void *pathname;
Packit Service 1d0348
	time_t sec;
Packit Service 1d0348
	long nsec;
Packit Service 1d0348
Packit Service 1d0348
	/*
Packit Service 1d0348
	 * If this file/dir is excluded by a time comparison, skip it.
Packit Service 1d0348
	 */
Packit Service 1d0348
	if (a->newer_ctime_filter) {
Packit Service 1d0348
		/* If ctime is not set, use mtime instead. */
Packit Service 1d0348
		if (archive_entry_ctime_is_set(entry))
Packit Service 1d0348
			sec = archive_entry_ctime(entry);
Packit Service 1d0348
		else
Packit Service 1d0348
			sec = archive_entry_mtime(entry);
Packit Service 1d0348
		if (sec < a->newer_ctime_sec)
Packit Service 1d0348
			return (1); /* Too old, skip it. */
Packit Service 1d0348
		if (sec == a->newer_ctime_sec) {
Packit Service 1d0348
			if (archive_entry_ctime_is_set(entry))
Packit Service 1d0348
				nsec = archive_entry_ctime_nsec(entry);
Packit Service 1d0348
			else
Packit Service 1d0348
				nsec = archive_entry_mtime_nsec(entry);
Packit Service 1d0348
			if (nsec < a->newer_ctime_nsec)
Packit Service 1d0348
				return (1); /* Too old, skip it. */
Packit Service 1d0348
			if (nsec == a->newer_ctime_nsec &&
Packit Service 1d0348
			    (a->newer_ctime_filter & ARCHIVE_MATCH_EQUAL)
Packit Service 1d0348
			      == 0)
Packit Service 1d0348
				return (1); /* Equal, skip it. */
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	if (a->older_ctime_filter) {
Packit Service 1d0348
		/* If ctime is not set, use mtime instead. */
Packit Service 1d0348
		if (archive_entry_ctime_is_set(entry))
Packit Service 1d0348
			sec = archive_entry_ctime(entry);
Packit Service 1d0348
		else
Packit Service 1d0348
			sec = archive_entry_mtime(entry);
Packit Service 1d0348
		if (sec > a->older_ctime_sec)
Packit Service 1d0348
			return (1); /* Too new, skip it. */
Packit Service 1d0348
		if (sec == a->older_ctime_sec) {
Packit Service 1d0348
			if (archive_entry_ctime_is_set(entry))
Packit Service 1d0348
				nsec = archive_entry_ctime_nsec(entry);
Packit Service 1d0348
			else
Packit Service 1d0348
				nsec = archive_entry_mtime_nsec(entry);
Packit Service 1d0348
			if (nsec > a->older_ctime_nsec)
Packit Service 1d0348
				return (1); /* Too new, skip it. */
Packit Service 1d0348
			if (nsec == a->older_ctime_nsec &&
Packit Service 1d0348
			    (a->older_ctime_filter & ARCHIVE_MATCH_EQUAL)
Packit Service 1d0348
			      == 0)
Packit Service 1d0348
				return (1); /* Equal, skip it. */
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	if (a->newer_mtime_filter) {
Packit Service 1d0348
		sec = archive_entry_mtime(entry);
Packit Service 1d0348
		if (sec < a->newer_mtime_sec)
Packit Service 1d0348
			return (1); /* Too old, skip it. */
Packit Service 1d0348
		if (sec == a->newer_mtime_sec) {
Packit Service 1d0348
			nsec = archive_entry_mtime_nsec(entry);
Packit Service 1d0348
			if (nsec < a->newer_mtime_nsec)
Packit Service 1d0348
				return (1); /* Too old, skip it. */
Packit Service 1d0348
			if (nsec == a->newer_mtime_nsec &&
Packit Service 1d0348
			    (a->newer_mtime_filter & ARCHIVE_MATCH_EQUAL)
Packit Service 1d0348
			       == 0)
Packit Service 1d0348
				return (1); /* Equal, skip it. */
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	if (a->older_mtime_filter) {
Packit Service 1d0348
		sec = archive_entry_mtime(entry);
Packit Service 1d0348
		if (sec > a->older_mtime_sec)
Packit Service 1d0348
			return (1); /* Too new, skip it. */
Packit Service 1d0348
		nsec = archive_entry_mtime_nsec(entry);
Packit Service 1d0348
		if (sec == a->older_mtime_sec) {
Packit Service 1d0348
			if (nsec > a->older_mtime_nsec)
Packit Service 1d0348
				return (1); /* Too new, skip it. */
Packit Service 1d0348
			if (nsec == a->older_mtime_nsec &&
Packit Service 1d0348
			    (a->older_mtime_filter & ARCHIVE_MATCH_EQUAL)
Packit Service 1d0348
			       == 0)
Packit Service 1d0348
				return (1); /* Equal, skip it. */
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* If there is no exclusion list, include the file. */
Packit Service 1d0348
	if (a->exclusion_entry_list.count == 0)
Packit Service 1d0348
		return (0);
Packit Service 1d0348
Packit Service 1d0348
#if defined(_WIN32) && !defined(__CYGWIN__)
Packit Service 1d0348
	pathname = archive_entry_pathname_w(entry);
Packit Service 1d0348
	a->exclusion_tree.rbt_ops = &rb_ops_wcs;
Packit Service 1d0348
#else
Packit Service 1d0348
	(void)rb_ops_wcs;
Packit Service 1d0348
	pathname = archive_entry_pathname(entry);
Packit Service 1d0348
	a->exclusion_tree.rbt_ops = &rb_ops_mbs;
Packit Service 1d0348
#endif
Packit Service 1d0348
	if (pathname == NULL)
Packit Service 1d0348
		return (0);
Packit Service 1d0348
Packit Service 1d0348
	f = (struct match_file *)__archive_rb_tree_find_node(
Packit Service 1d0348
		&(a->exclusion_tree), pathname);
Packit Service 1d0348
	/* If the file wasn't rejected, include it. */
Packit Service 1d0348
	if (f == NULL)
Packit Service 1d0348
		return (0);
Packit Service 1d0348
Packit Service 1d0348
	if (f->flag & ARCHIVE_MATCH_CTIME) {
Packit Service 1d0348
		sec = archive_entry_ctime(entry);
Packit Service 1d0348
		if (f->ctime_sec > sec) {
Packit Service 1d0348
			if (f->flag & ARCHIVE_MATCH_OLDER)
Packit Service 1d0348
				return (1);
Packit Service 1d0348
		} else if (f->ctime_sec < sec) {
Packit Service 1d0348
			if (f->flag & ARCHIVE_MATCH_NEWER)
Packit Service 1d0348
				return (1);
Packit Service 1d0348
		} else {
Packit Service 1d0348
			nsec = archive_entry_ctime_nsec(entry);
Packit Service 1d0348
			if (f->ctime_nsec > nsec) {
Packit Service 1d0348
				if (f->flag & ARCHIVE_MATCH_OLDER)
Packit Service 1d0348
					return (1);
Packit Service 1d0348
			} else if (f->ctime_nsec < nsec) {
Packit Service 1d0348
				if (f->flag & ARCHIVE_MATCH_NEWER)
Packit Service 1d0348
					return (1);
Packit Service 1d0348
			} else if (f->flag & ARCHIVE_MATCH_EQUAL)
Packit Service 1d0348
				return (1);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	if (f->flag & ARCHIVE_MATCH_MTIME) {
Packit Service 1d0348
		sec = archive_entry_mtime(entry);
Packit Service 1d0348
		if (f->mtime_sec > sec) {
Packit Service 1d0348
			if (f->flag & ARCHIVE_MATCH_OLDER)
Packit Service 1d0348
				return (1);
Packit Service 1d0348
		} else if (f->mtime_sec < sec) {
Packit Service 1d0348
			if (f->flag & ARCHIVE_MATCH_NEWER)
Packit Service 1d0348
				return (1);
Packit Service 1d0348
		} else {
Packit Service 1d0348
			nsec = archive_entry_mtime_nsec(entry);
Packit Service 1d0348
			if (f->mtime_nsec > nsec) {
Packit Service 1d0348
				if (f->flag & ARCHIVE_MATCH_OLDER)
Packit Service 1d0348
					return (1);
Packit Service 1d0348
			} else if (f->mtime_nsec < nsec) {
Packit Service 1d0348
				if (f->flag & ARCHIVE_MATCH_NEWER)
Packit Service 1d0348
					return (1);
Packit Service 1d0348
			} else if (f->flag & ARCHIVE_MATCH_EQUAL)
Packit Service 1d0348
				return (1);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Utility functions to manage inclusion owners
Packit Service 1d0348
 */
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_uid(struct archive *_a, la_int64_t uid)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_uid");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	return (add_owner_id(a, &(a->inclusion_uids), uid));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_gid(struct archive *_a, la_int64_t gid)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_gid");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	return (add_owner_id(a, &(a->inclusion_gids), gid));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_uname(struct archive *_a, const char *uname)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_uname");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	return (add_owner_name(a, &(a->inclusion_unames), 1, uname));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_uname_w(struct archive *_a, const wchar_t *uname)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_uname_w");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	return (add_owner_name(a, &(a->inclusion_unames), 0, uname));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_gname(struct archive *_a, const char *gname)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_gname");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	return (add_owner_name(a, &(a->inclusion_gnames), 1, gname));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_include_gname_w(struct archive *_a, const wchar_t *gname)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_include_gname_w");
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	return (add_owner_name(a, &(a->inclusion_gnames), 0, gname));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Test function for owner(uid, gid, uname, gname).
Packit Service 1d0348
 *
Packit Service 1d0348
 * Returns 1 if archive entry is excluded.
Packit Service 1d0348
 * Returns 0 if archive entry is not excluded.
Packit Service 1d0348
 * Returns <0 if something error happened.
Packit Service 1d0348
 */
Packit Service 1d0348
int
Packit Service 1d0348
archive_match_owner_excluded(struct archive *_a,
Packit Service 1d0348
    struct archive_entry *entry)
Packit Service 1d0348
{
Packit Service 1d0348
	struct archive_match *a;
Packit Service 1d0348
Packit Service 1d0348
	archive_check_magic(_a, ARCHIVE_MATCH_MAGIC,
Packit Service 1d0348
	    ARCHIVE_STATE_NEW, "archive_match_id_excluded_ae");
Packit Service 1d0348
Packit Service 1d0348
	a = (struct archive_match *)_a;
Packit Service 1d0348
	if (entry == NULL) {
Packit Service 1d0348
		archive_set_error(&(a->archive), EINVAL, "entry is NULL");
Packit Service 1d0348
		return (ARCHIVE_FAILED);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* If we don't have inclusion id set at all, the entry is always
Packit Service 1d0348
	 * not excluded. */
Packit Service 1d0348
	if ((a->setflag & ID_IS_SET) == 0)
Packit Service 1d0348
		return (0);
Packit Service 1d0348
	return (owner_excluded(a, entry));
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
add_owner_id(struct archive_match *a, struct id_array *ids, int64_t id)
Packit Service 1d0348
{
Packit Service 1d0348
	unsigned i;
Packit Service 1d0348
Packit Service 1d0348
	if (ids->count + 1 >= ids->size) {
Packit Service 1d0348
		void *p;
Packit Service 1d0348
Packit Service 1d0348
		if (ids->size == 0)
Packit Service 1d0348
			ids->size = 8;
Packit Service 1d0348
		else
Packit Service 1d0348
			ids->size *= 2;
Packit Service 1d0348
		p = realloc(ids->ids, sizeof(*ids->ids) * ids->size);
Packit Service 1d0348
		if (p == NULL)
Packit Service 1d0348
			return (error_nomem(a));
Packit Service 1d0348
		ids->ids = (int64_t *)p;
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* Find an insert point. */
Packit Service 1d0348
	for (i = 0; i < ids->count; i++) {
Packit Service 1d0348
		if (ids->ids[i] >= id)
Packit Service 1d0348
			break;
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	/* Add owner id. */
Packit Service 1d0348
	if (i == ids->count)
Packit Service 1d0348
		ids->ids[ids->count++] = id;
Packit Service 1d0348
	else if (ids->ids[i] != id) {
Packit Service 1d0348
		memmove(&(ids->ids[i+1]), &(ids->ids[i]),
Packit Service 1d0348
		    (ids->count - i) * sizeof(ids->ids[0]));
Packit Service 1d0348
		ids->ids[i] = id;
Packit Service 1d0348
		ids->count++;
Packit Service 1d0348
	}
Packit Service 1d0348
	a->setflag |= ID_IS_SET;
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
match_owner_id(struct id_array *ids, int64_t id)
Packit Service 1d0348
{
Packit Service 1d0348
	unsigned b, m, t;
Packit Service 1d0348
Packit Service 1d0348
	t = 0;
Packit Service 1d0348
	b = (unsigned)ids->count;
Packit Service 1d0348
	while (t < b) {
Packit Service 1d0348
		m = (t + b)>>1;
Packit Service 1d0348
		if (ids->ids[m] == id)
Packit Service 1d0348
			return (1);
Packit Service 1d0348
		if (ids->ids[m] < id)
Packit Service 1d0348
			t = m + 1;
Packit Service 1d0348
		else
Packit Service 1d0348
			b = m;
Packit Service 1d0348
	}
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
static int
Packit Service 1d0348
add_owner_name(struct archive_match *a, struct match_list *list,
Packit Service 1d0348
    int mbs, const void *name)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match *match;
Packit Service 1d0348
Packit Service 1d0348
	match = calloc(1, sizeof(*match));
Packit Service 1d0348
	if (match == NULL)
Packit Service 1d0348
		return (error_nomem(a));
Packit Service 1d0348
	if (mbs)
Packit Service 1d0348
		archive_mstring_copy_mbs(&(match->pattern), name);
Packit Service 1d0348
	else
Packit Service 1d0348
		archive_mstring_copy_wcs(&(match->pattern), name);
Packit Service 1d0348
	match_list_add(list, match);
Packit Service 1d0348
	a->setflag |= ID_IS_SET;
Packit Service 1d0348
	return (ARCHIVE_OK);
Packit Service 1d0348
}
Packit Service 1d0348
Packit Service 1d0348
#if !defined(_WIN32) || defined(__CYGWIN__)
Packit Service 1d0348
static int
Packit Service 1d0348
match_owner_name_mbs(struct archive_match *a, struct match_list *list,
Packit Service 1d0348
    const char *name)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match *m;
Packit Service 1d0348
	const char *p;
Packit Service 1d0348
Packit Service 1d0348
	if (name == NULL || *name == '\0')
Packit Service 1d0348
		return (0);
Packit Service 1d0348
	for (m = list->first; m; m = m->next) {
Packit Service 1d0348
		if (archive_mstring_get_mbs(&(a->archive), &(m->pattern), &p)
Packit Service 1d0348
		    < 0 && errno == ENOMEM)
Packit Service 1d0348
			return (error_nomem(a));
Packit Service 1d0348
		if (p != NULL && strcmp(p, name) == 0) {
Packit Service 1d0348
			m->matches++;
Packit Service 1d0348
			return (1);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}
Packit Service 1d0348
#else
Packit Service 1d0348
static int
Packit Service 1d0348
match_owner_name_wcs(struct archive_match *a, struct match_list *list,
Packit Service 1d0348
    const wchar_t *name)
Packit Service 1d0348
{
Packit Service 1d0348
	struct match *m;
Packit Service 1d0348
	const wchar_t *p;
Packit Service 1d0348
Packit Service 1d0348
	if (name == NULL || *name == L'\0')
Packit Service 1d0348
		return (0);
Packit Service 1d0348
	for (m = list->first; m; m = m->next) {
Packit Service 1d0348
		if (archive_mstring_get_wcs(&(a->archive), &(m->pattern), &p)
Packit Service 1d0348
		    < 0 && errno == ENOMEM)
Packit Service 1d0348
			return (error_nomem(a));
Packit Service 1d0348
		if (p != NULL && wcscmp(p, name) == 0) {
Packit Service 1d0348
			m->matches++;
Packit Service 1d0348
			return (1);
Packit Service 1d0348
		}
Packit Service 1d0348
	}
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}
Packit Service 1d0348
#endif
Packit Service 1d0348
Packit Service 1d0348
/*
Packit Service 1d0348
 * Test if entry is excluded by uid, gid, uname or gname.
Packit Service 1d0348
 */
Packit Service 1d0348
static int
Packit Service 1d0348
owner_excluded(struct archive_match *a, struct archive_entry *entry)
Packit Service 1d0348
{
Packit Service 1d0348
	int r;
Packit Service 1d0348
Packit Service 1d0348
	if (a->inclusion_uids.count) {
Packit Service 1d0348
		if (!match_owner_id(&(a->inclusion_uids),
Packit Service 1d0348
		    archive_entry_uid(entry)))
Packit Service 1d0348
			return (1);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	if (a->inclusion_gids.count) {
Packit Service 1d0348
		if (!match_owner_id(&(a->inclusion_gids),
Packit Service 1d0348
		    archive_entry_gid(entry)))
Packit Service 1d0348
			return (1);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	if (a->inclusion_unames.count) {
Packit Service 1d0348
#if defined(_WIN32) && !defined(__CYGWIN__)
Packit Service 1d0348
		r = match_owner_name_wcs(a, &(a->inclusion_unames),
Packit Service 1d0348
			archive_entry_uname_w(entry));
Packit Service 1d0348
#else
Packit Service 1d0348
		r = match_owner_name_mbs(a, &(a->inclusion_unames),
Packit Service 1d0348
			archive_entry_uname(entry));
Packit Service 1d0348
#endif
Packit Service 1d0348
		if (!r)
Packit Service 1d0348
			return (1);
Packit Service 1d0348
		else if (r < 0)
Packit Service 1d0348
			return (r);
Packit Service 1d0348
	}
Packit Service 1d0348
Packit Service 1d0348
	if (a->inclusion_gnames.count) {
Packit Service 1d0348
#if defined(_WIN32) && !defined(__CYGWIN__)
Packit Service 1d0348
		r = match_owner_name_wcs(a, &(a->inclusion_gnames),
Packit Service 1d0348
			archive_entry_gname_w(entry));
Packit Service 1d0348
#else
Packit Service 1d0348
		r = match_owner_name_mbs(a, &(a->inclusion_gnames),
Packit Service 1d0348
			archive_entry_gname(entry));
Packit Service 1d0348
#endif
Packit Service 1d0348
		if (!r)
Packit Service 1d0348
			return (1);
Packit Service 1d0348
		else if (r < 0)
Packit Service 1d0348
			return (r);
Packit Service 1d0348
	}
Packit Service 1d0348
	return (0);
Packit Service 1d0348
}
Packit Service 1d0348