Blame libarchive/archive_read_support_format_mtree.c

Packit 08bd4c
/*-
Packit 08bd4c
 * Copyright (c) 2003-2007 Tim Kientzle
Packit 08bd4c
 * Copyright (c) 2008 Joerg Sonnenberger
Packit 08bd4c
 * Copyright (c) 2011-2012 Michihiro NAKAJIMA
Packit 08bd4c
 * All rights reserved.
Packit 08bd4c
 *
Packit 08bd4c
 * Redistribution and use in source and binary forms, with or without
Packit 08bd4c
 * modification, are permitted provided that the following conditions
Packit 08bd4c
 * are met:
Packit 08bd4c
 * 1. Redistributions of source code must retain the above copyright
Packit 08bd4c
 *    notice, this list of conditions and the following disclaimer.
Packit 08bd4c
 * 2. Redistributions in binary form must reproduce the above copyright
Packit 08bd4c
 *    notice, this list of conditions and the following disclaimer in the
Packit 08bd4c
 *    documentation and/or other materials provided with the distribution.
Packit 08bd4c
 *
Packit 08bd4c
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
Packit 08bd4c
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
Packit 08bd4c
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
Packit 08bd4c
 * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
Packit 08bd4c
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
Packit 08bd4c
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
Packit 08bd4c
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
Packit 08bd4c
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit 08bd4c
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
Packit 08bd4c
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit 08bd4c
 */
Packit 08bd4c
Packit 08bd4c
#include "archive_platform.h"
Packit 08bd4c
__FBSDID("$FreeBSD: head/lib/libarchive/archive_read_support_format_mtree.c 201165 2009-12-29 05:52:13Z kientzle $");
Packit 08bd4c
Packit 08bd4c
#ifdef HAVE_SYS_STAT_H
Packit 08bd4c
#include <sys/stat.h>
Packit 08bd4c
#endif
Packit 08bd4c
#ifdef HAVE_ERRNO_H
Packit 08bd4c
#include <errno.h>
Packit 08bd4c
#endif
Packit 08bd4c
#ifdef HAVE_FCNTL_H
Packit 08bd4c
#include <fcntl.h>
Packit 08bd4c
#endif
Packit 08bd4c
#include <stddef.h>
Packit 08bd4c
/* #include <stdint.h> */ /* See archive_platform.h */
Packit 08bd4c
#ifdef HAVE_STDLIB_H
Packit 08bd4c
#include <stdlib.h>
Packit 08bd4c
#endif
Packit 08bd4c
#ifdef HAVE_STRING_H
Packit 08bd4c
#include <string.h>
Packit 08bd4c
#endif
Packit 08bd4c
Packit 08bd4c
#include "archive.h"
Packit 08bd4c
#include "archive_entry.h"
Packit 08bd4c
#include "archive_private.h"
Packit 08bd4c
#include "archive_read_private.h"
Packit 08bd4c
#include "archive_string.h"
Packit 08bd4c
#include "archive_pack_dev.h"
Packit 08bd4c
Packit 08bd4c
#ifndef O_BINARY
Packit 08bd4c
#define	O_BINARY 0
Packit 08bd4c
#endif
Packit 08bd4c
#ifndef O_CLOEXEC
Packit 08bd4c
#define O_CLOEXEC	0
Packit 08bd4c
#endif
Packit 08bd4c
Packit 08bd4c
#define	MTREE_HAS_DEVICE	0x0001
Packit 08bd4c
#define	MTREE_HAS_FFLAGS	0x0002
Packit 08bd4c
#define	MTREE_HAS_GID		0x0004
Packit 08bd4c
#define	MTREE_HAS_GNAME		0x0008
Packit 08bd4c
#define	MTREE_HAS_MTIME		0x0010
Packit 08bd4c
#define	MTREE_HAS_NLINK		0x0020
Packit 08bd4c
#define	MTREE_HAS_PERM		0x0040
Packit 08bd4c
#define	MTREE_HAS_SIZE		0x0080
Packit 08bd4c
#define	MTREE_HAS_TYPE		0x0100
Packit 08bd4c
#define	MTREE_HAS_UID		0x0200
Packit 08bd4c
#define	MTREE_HAS_UNAME		0x0400
Packit 08bd4c
Packit 08bd4c
#define	MTREE_HAS_OPTIONAL	0x0800
Packit 08bd4c
#define	MTREE_HAS_NOCHANGE	0x1000 /* FreeBSD specific */
Packit 08bd4c
Packit 08bd4c
#define	MTREE_HASHTABLE_SIZE 1024
Packit 08bd4c
Packit 08bd4c
struct mtree_option {
Packit 08bd4c
	struct mtree_option *next;
Packit 08bd4c
	char *value;
Packit 08bd4c
};
Packit 08bd4c
Packit 08bd4c
struct mtree_entry {
Packit 08bd4c
	struct mtree_entry *next;
Packit 08bd4c
	struct mtree_option *options;
Packit 08bd4c
	char *name;
Packit 08bd4c
	char full;
Packit 08bd4c
	char used;
Packit 08bd4c
	unsigned int name_hash;
Packit 08bd4c
	struct mtree_entry *hashtable_next;
Packit 08bd4c
};
Packit 08bd4c
Packit 08bd4c
struct mtree {
Packit 08bd4c
	struct archive_string	 line;
Packit 08bd4c
	size_t			 buffsize;
Packit 08bd4c
	char			*buff;
Packit 08bd4c
	int64_t			 offset;
Packit 08bd4c
	int			 fd;
Packit 08bd4c
	int			 archive_format;
Packit 08bd4c
	const char		*archive_format_name;
Packit 08bd4c
	struct mtree_entry	*entries;
Packit 08bd4c
	struct mtree_entry	*this_entry;
Packit 08bd4c
	struct mtree_entry	*entry_hashtable[MTREE_HASHTABLE_SIZE];
Packit 08bd4c
	struct archive_string	 current_dir;
Packit 08bd4c
	struct archive_string	 contents_name;
Packit 08bd4c
Packit 08bd4c
	struct archive_entry_linkresolver *resolver;
Packit 08bd4c
Packit 08bd4c
	int64_t			 cur_size;
Packit 08bd4c
	char checkfs;
Packit 08bd4c
};
Packit 08bd4c
Packit 08bd4c
static int	bid_keycmp(const char *, const char *, ssize_t);
Packit 08bd4c
static int	cleanup(struct archive_read *);
Packit 08bd4c
static int	detect_form(struct archive_read *, int *);
Packit 08bd4c
static unsigned int	hash(const char *);
Packit 08bd4c
static int	mtree_bid(struct archive_read *, int);
Packit 08bd4c
static int	parse_file(struct archive_read *, struct archive_entry *,
Packit 08bd4c
		    struct mtree *, struct mtree_entry *, int *);
Packit 08bd4c
static void	parse_escapes(char *, struct mtree_entry *);
Packit 08bd4c
static int	parse_line(struct archive_read *, struct archive_entry *,
Packit 08bd4c
		    struct mtree *, struct mtree_entry *, int *);
Packit 08bd4c
static int	parse_keyword(struct archive_read *, struct mtree *,
Packit 08bd4c
		    struct archive_entry *, struct mtree_option *, int *);
Packit 08bd4c
static int	read_data(struct archive_read *a,
Packit 08bd4c
		    const void **buff, size_t *size, int64_t *offset);
Packit 08bd4c
static ssize_t	readline(struct archive_read *, struct mtree *, char **, ssize_t);
Packit 08bd4c
static int	skip(struct archive_read *a);
Packit 08bd4c
static int	read_header(struct archive_read *,
Packit 08bd4c
		    struct archive_entry *);
Packit 08bd4c
static int64_t	mtree_atol(char **, int base);
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * There's no standard for TIME_T_MAX/TIME_T_MIN.  So we compute them
Packit 08bd4c
 * here.  TODO: Move this to configure time, but be careful
Packit 08bd4c
 * about cross-compile environments.
Packit 08bd4c
 */
Packit 08bd4c
static int64_t
Packit 08bd4c
get_time_t_max(void)
Packit 08bd4c
{
Packit 08bd4c
#if defined(TIME_T_MAX)
Packit 08bd4c
	return TIME_T_MAX;
Packit 08bd4c
#else
Packit 08bd4c
	/* ISO C allows time_t to be a floating-point type,
Packit 08bd4c
	   but POSIX requires an integer type.  The following
Packit 08bd4c
	   should work on any system that follows the POSIX
Packit 08bd4c
	   conventions. */
Packit 08bd4c
	if (((time_t)0) < ((time_t)-1)) {
Packit 08bd4c
		/* Time_t is unsigned */
Packit 08bd4c
		return (~(time_t)0);
Packit 08bd4c
	} else {
Packit 08bd4c
		/* Time_t is signed. */
Packit 08bd4c
		/* Assume it's the same as int64_t or int32_t */
Packit 08bd4c
		if (sizeof(time_t) == sizeof(int64_t)) {
Packit 08bd4c
			return (time_t)INT64_MAX;
Packit 08bd4c
		} else {
Packit 08bd4c
			return (time_t)INT32_MAX;
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
#endif
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int64_t
Packit 08bd4c
get_time_t_min(void)
Packit 08bd4c
{
Packit 08bd4c
#if defined(TIME_T_MIN)
Packit 08bd4c
	return TIME_T_MIN;
Packit 08bd4c
#else
Packit 08bd4c
	if (((time_t)0) < ((time_t)-1)) {
Packit 08bd4c
		/* Time_t is unsigned */
Packit 08bd4c
		return (time_t)0;
Packit 08bd4c
	} else {
Packit 08bd4c
		/* Time_t is signed. */
Packit 08bd4c
		if (sizeof(time_t) == sizeof(int64_t)) {
Packit 08bd4c
			return (time_t)INT64_MIN;
Packit 08bd4c
		} else {
Packit 08bd4c
			return (time_t)INT32_MIN;
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
#endif
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
archive_read_format_mtree_options(struct archive_read *a,
Packit 08bd4c
    const char *key, const char *val)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree *mtree;
Packit 08bd4c
Packit 08bd4c
	mtree = (struct mtree *)(a->format->data);
Packit 08bd4c
	if (strcmp(key, "checkfs")  == 0) {
Packit 08bd4c
		/* Allows to read information missing from the mtree from the file system */
Packit 08bd4c
		if (val == NULL || val[0] == 0) {
Packit 08bd4c
			mtree->checkfs = 0;
Packit 08bd4c
		} else {
Packit 08bd4c
			mtree->checkfs = 1;
Packit 08bd4c
		}
Packit 08bd4c
		return (ARCHIVE_OK);
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	/* Note: The "warn" return is just to inform the options
Packit 08bd4c
	 * supervisor that we didn't handle it.  It will generate
Packit 08bd4c
	 * a suitable error if no one used this option. */
Packit 08bd4c
	return (ARCHIVE_WARN);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static void
Packit 08bd4c
free_options(struct mtree_option *head)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree_option *next;
Packit 08bd4c
Packit 08bd4c
	for (; head != NULL; head = next) {
Packit 08bd4c
		next = head->next;
Packit 08bd4c
		free(head->value);
Packit 08bd4c
		free(head);
Packit 08bd4c
	}
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
int
Packit 08bd4c
archive_read_support_format_mtree(struct archive *_a)
Packit 08bd4c
{
Packit 08bd4c
	struct archive_read *a = (struct archive_read *)_a;
Packit 08bd4c
	struct mtree *mtree;
Packit 08bd4c
	int r;
Packit 08bd4c
Packit 08bd4c
	archive_check_magic(_a, ARCHIVE_READ_MAGIC,
Packit 08bd4c
	    ARCHIVE_STATE_NEW, "archive_read_support_format_mtree");
Packit 08bd4c
Packit 08bd4c
	mtree = (struct mtree *)calloc(1, sizeof(*mtree));
Packit 08bd4c
	if (mtree == NULL) {
Packit 08bd4c
		archive_set_error(&a->archive, ENOMEM,
Packit 08bd4c
		    "Can't allocate mtree data");
Packit 08bd4c
		return (ARCHIVE_FATAL);
Packit 08bd4c
	}
Packit 08bd4c
	mtree->fd = -1;
Packit 08bd4c
Packit 08bd4c
	r = __archive_read_register_format(a, mtree, "mtree",
Packit 08bd4c
           mtree_bid, archive_read_format_mtree_options, read_header, read_data, skip, NULL, cleanup, NULL, NULL);
Packit 08bd4c
Packit 08bd4c
	if (r != ARCHIVE_OK)
Packit 08bd4c
		free(mtree);
Packit 08bd4c
	return (ARCHIVE_OK);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
cleanup(struct archive_read *a)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree *mtree;
Packit 08bd4c
	struct mtree_entry *p, *q;
Packit 08bd4c
Packit 08bd4c
	mtree = (struct mtree *)(a->format->data);
Packit 08bd4c
Packit 08bd4c
	p = mtree->entries;
Packit 08bd4c
	while (p != NULL) {
Packit 08bd4c
		q = p->next;
Packit 08bd4c
		free(p->name);
Packit 08bd4c
		free_options(p->options);
Packit 08bd4c
		free(p);
Packit 08bd4c
		p = q;
Packit 08bd4c
	}
Packit 08bd4c
	archive_string_free(&mtree->line);
Packit 08bd4c
	archive_string_free(&mtree->current_dir);
Packit 08bd4c
	archive_string_free(&mtree->contents_name);
Packit 08bd4c
	archive_entry_linkresolver_free(mtree->resolver);
Packit 08bd4c
Packit 08bd4c
	free(mtree->buff);
Packit 08bd4c
	free(mtree);
Packit 08bd4c
	(a->format->data) = NULL;
Packit 08bd4c
	return (ARCHIVE_OK);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static ssize_t
Packit 08bd4c
get_line_size(const char *b, ssize_t avail, ssize_t *nlsize)
Packit 08bd4c
{
Packit 08bd4c
	ssize_t len;
Packit 08bd4c
Packit 08bd4c
	len = 0;
Packit 08bd4c
	while (len < avail) {
Packit 08bd4c
		switch (*b) {
Packit 08bd4c
		case '\0':/* Non-ascii character or control character. */
Packit 08bd4c
			if (nlsize != NULL)
Packit 08bd4c
				*nlsize = 0;
Packit 08bd4c
			return (-1);
Packit 08bd4c
		case '\r':
Packit 08bd4c
			if (avail-len > 1 && b[1] == '\n') {
Packit 08bd4c
				if (nlsize != NULL)
Packit 08bd4c
					*nlsize = 2;
Packit 08bd4c
				return (len+2);
Packit 08bd4c
			}
Packit 08bd4c
			/* FALL THROUGH */
Packit 08bd4c
		case '\n':
Packit 08bd4c
			if (nlsize != NULL)
Packit 08bd4c
				*nlsize = 1;
Packit 08bd4c
			return (len+1);
Packit 08bd4c
		default:
Packit 08bd4c
			b++;
Packit 08bd4c
			len++;
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
	if (nlsize != NULL)
Packit 08bd4c
		*nlsize = 0;
Packit 08bd4c
	return (avail);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 *  <---------------- ravail --------------------->
Packit 08bd4c
 *  <-- diff ------> <---  avail ----------------->
Packit 08bd4c
 *                   <---- len ----------->
Packit 08bd4c
 * | Previous lines | line being parsed  nl extra |
Packit 08bd4c
 *                  ^
Packit 08bd4c
 *                  b
Packit 08bd4c
 *
Packit 08bd4c
 */
Packit 08bd4c
static ssize_t
Packit 08bd4c
next_line(struct archive_read *a,
Packit 08bd4c
    const char **b, ssize_t *avail, ssize_t *ravail, ssize_t *nl)
Packit 08bd4c
{
Packit 08bd4c
	ssize_t len;
Packit 08bd4c
	int quit;
Packit 08bd4c
	
Packit 08bd4c
	quit = 0;
Packit 08bd4c
	if (*avail == 0) {
Packit 08bd4c
		*nl = 0;
Packit 08bd4c
		len = 0;
Packit 08bd4c
	} else
Packit 08bd4c
		len = get_line_size(*b, *avail, nl);
Packit 08bd4c
	/*
Packit 08bd4c
	 * Read bytes more while it does not reach the end of line.
Packit 08bd4c
	 */
Packit 08bd4c
	while (*nl == 0 && len == *avail && !quit) {
Packit 08bd4c
		ssize_t diff = *ravail - *avail;
Packit 08bd4c
		size_t nbytes_req = (*ravail+1023) & ~1023U;
Packit 08bd4c
		ssize_t tested;
Packit 08bd4c
Packit 08bd4c
		/* Increase reading bytes if it is not enough to at least
Packit 08bd4c
		 * new two lines. */
Packit 08bd4c
		if (nbytes_req < (size_t)*ravail + 160)
Packit 08bd4c
			nbytes_req <<= 1;
Packit 08bd4c
Packit 08bd4c
		*b = __archive_read_ahead(a, nbytes_req, avail);
Packit 08bd4c
		if (*b == NULL) {
Packit 08bd4c
			if (*ravail >= *avail)
Packit 08bd4c
				return (0);
Packit 08bd4c
			/* Reading bytes reaches the end of file. */
Packit 08bd4c
			*b = __archive_read_ahead(a, *avail, avail);
Packit 08bd4c
			quit = 1;
Packit 08bd4c
		}
Packit 08bd4c
		*ravail = *avail;
Packit 08bd4c
		*b += diff;
Packit 08bd4c
		*avail -= diff;
Packit 08bd4c
		tested = len;/* Skip some bytes we already determinated. */
Packit 08bd4c
		len = get_line_size(*b + len, *avail - len, nl);
Packit 08bd4c
		if (len >= 0)
Packit 08bd4c
			len += tested;
Packit 08bd4c
	}
Packit 08bd4c
	return (len);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Compare characters with a mtree keyword.
Packit 08bd4c
 * Returns the length of a mtree keyword if matched.
Packit 08bd4c
 * Returns 0 if not matched.
Packit 08bd4c
 */
Packit 08bd4c
static int
Packit 08bd4c
bid_keycmp(const char *p, const char *key, ssize_t len)
Packit 08bd4c
{
Packit 08bd4c
	int match_len = 0;
Packit 08bd4c
Packit 08bd4c
	while (len > 0 && *p && *key) {
Packit 08bd4c
		if (*p == *key) {
Packit 08bd4c
			--len;
Packit 08bd4c
			++p;
Packit 08bd4c
			++key;
Packit 08bd4c
			++match_len;
Packit 08bd4c
			continue;
Packit 08bd4c
		}
Packit 08bd4c
		return (0);/* Not match */
Packit 08bd4c
	}
Packit 08bd4c
	if (*key != '\0')
Packit 08bd4c
		return (0);/* Not match */
Packit 08bd4c
Packit 08bd4c
	/* A following character should be specified characters */
Packit 08bd4c
	if (p[0] == '=' || p[0] == ' ' || p[0] == '\t' ||
Packit 08bd4c
	    p[0] == '\n' || p[0] == '\r' ||
Packit 08bd4c
	   (p[0] == '\\' && (p[1] == '\n' || p[1] == '\r')))
Packit 08bd4c
		return (match_len);
Packit 08bd4c
	return (0);/* Not match */
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Test whether the characters 'p' has is mtree keyword.
Packit 08bd4c
 * Returns the length of a detected keyword.
Packit 08bd4c
 * Returns 0 if any keywords were not found.
Packit 08bd4c
 */
Packit 08bd4c
static int
Packit 08bd4c
bid_keyword(const char *p,  ssize_t len)
Packit 08bd4c
{
Packit 08bd4c
	static const char * const keys_c[] = {
Packit 08bd4c
		"content", "contents", "cksum", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_df[] = {
Packit 08bd4c
		"device", "flags", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_g[] = {
Packit 08bd4c
		"gid", "gname", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_il[] = {
Packit 08bd4c
		"ignore", "inode", "link", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_m[] = {
Packit 08bd4c
		"md5", "md5digest", "mode", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_no[] = {
Packit 08bd4c
		"nlink", "nochange", "optional", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_r[] = {
Packit 08bd4c
		"resdevice", "rmd160", "rmd160digest", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_s[] = {
Packit 08bd4c
		"sha1", "sha1digest",
Packit 08bd4c
		"sha256", "sha256digest",
Packit 08bd4c
		"sha384", "sha384digest",
Packit 08bd4c
		"sha512", "sha512digest",
Packit 08bd4c
		"size", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_t[] = {
Packit 08bd4c
		"tags", "time", "type", NULL
Packit 08bd4c
	};
Packit 08bd4c
	static const char * const keys_u[] = {
Packit 08bd4c
		"uid", "uname",	NULL
Packit 08bd4c
	};
Packit 08bd4c
	const char * const *keys;
Packit 08bd4c
	int i;
Packit 08bd4c
Packit 08bd4c
	switch (*p) {
Packit 08bd4c
	case 'c': keys = keys_c; break;
Packit 08bd4c
	case 'd': case 'f': keys = keys_df; break;
Packit 08bd4c
	case 'g': keys = keys_g; break;
Packit 08bd4c
	case 'i': case 'l': keys = keys_il; break;
Packit 08bd4c
	case 'm': keys = keys_m; break;
Packit 08bd4c
	case 'n': case 'o': keys = keys_no; break;
Packit 08bd4c
	case 'r': keys = keys_r; break;
Packit 08bd4c
	case 's': keys = keys_s; break;
Packit 08bd4c
	case 't': keys = keys_t; break;
Packit 08bd4c
	case 'u': keys = keys_u; break;
Packit 08bd4c
	default: return (0);/* Unknown key */
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	for (i = 0; keys[i] != NULL; i++) {
Packit 08bd4c
		int l = bid_keycmp(p, keys[i], len);
Packit 08bd4c
		if (l > 0)
Packit 08bd4c
			return (l);
Packit 08bd4c
	}
Packit 08bd4c
	return (0);/* Unknown key */
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Test whether there is a set of mtree keywords.
Packit 08bd4c
 * Returns the number of keyword.
Packit 08bd4c
 * Returns -1 if we got incorrect sequence.
Packit 08bd4c
 * This function expects a set of "<space characters>keyword=value".
Packit 08bd4c
 * When "unset" is specified, expects a set of "<space characters>keyword".
Packit 08bd4c
 */
Packit 08bd4c
static int
Packit 08bd4c
bid_keyword_list(const char *p,  ssize_t len, int unset, int last_is_path)
Packit 08bd4c
{
Packit 08bd4c
	int l;
Packit 08bd4c
	int keycnt = 0;
Packit 08bd4c
Packit 08bd4c
	while (len > 0 && *p) {
Packit 08bd4c
		int blank = 0;
Packit 08bd4c
Packit 08bd4c
		/* Test whether there are blank characters in the line. */
Packit 08bd4c
		while (len >0 && (*p == ' ' || *p == '\t')) {
Packit 08bd4c
			++p;
Packit 08bd4c
			--len;
Packit 08bd4c
			blank = 1;
Packit 08bd4c
		}
Packit 08bd4c
		if (*p == '\n' || *p == '\r')
Packit 08bd4c
			break;
Packit 08bd4c
		if (p[0] == '\\' && (p[1] == '\n' || p[1] == '\r'))
Packit 08bd4c
			break;
Packit 08bd4c
		if (!blank && !last_is_path) /* No blank character. */
Packit 08bd4c
			return (-1);
Packit 08bd4c
		if (last_is_path && len == 0)
Packit 08bd4c
				return (keycnt);
Packit 08bd4c
Packit 08bd4c
		if (unset) {
Packit 08bd4c
			l = bid_keycmp(p, "all", len);
Packit 08bd4c
			if (l > 0)
Packit 08bd4c
				return (1);
Packit 08bd4c
		}
Packit 08bd4c
		/* Test whether there is a correct key in the line. */
Packit 08bd4c
		l = bid_keyword(p, len);
Packit 08bd4c
		if (l == 0)
Packit 08bd4c
			return (-1);/* Unknown keyword was found. */
Packit 08bd4c
		p += l;
Packit 08bd4c
		len -= l;
Packit 08bd4c
		keycnt++;
Packit 08bd4c
Packit 08bd4c
		/* Skip value */
Packit 08bd4c
		if (*p == '=') {
Packit 08bd4c
			int value = 0;
Packit 08bd4c
			++p;
Packit 08bd4c
			--len;
Packit 08bd4c
			while (len > 0 && *p != ' ' && *p != '\t') {
Packit 08bd4c
				++p;
Packit 08bd4c
				--len;
Packit 08bd4c
				value = 1;
Packit 08bd4c
			}
Packit 08bd4c
			/* A keyword should have a its value unless
Packit 08bd4c
			 * "/unset" operation. */ 
Packit 08bd4c
			if (!unset && value == 0)
Packit 08bd4c
				return (-1);
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
	return (keycnt);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
bid_entry(const char *p, ssize_t len, ssize_t nl, int *last_is_path)
Packit 08bd4c
{
Packit 08bd4c
	int f = 0;
Packit 08bd4c
	static const unsigned char safe_char[256] = {
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 00 - 0F */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 10 - 1F */
Packit 08bd4c
		/* !"$%&'()*+,-./  EXCLUSION:( )(#) */
Packit 08bd4c
		0, 1, 1, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 20 - 2F */
Packit 08bd4c
		/* 0123456789:;<>?  EXCLUSION:(=) */
Packit 08bd4c
		1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, /* 30 - 3F */
Packit 08bd4c
		/* @ABCDEFGHIJKLMNO */
Packit 08bd4c
		1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 40 - 4F */
Packit 08bd4c
		/* PQRSTUVWXYZ[\]^_  */
Packit 08bd4c
		1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 50 - 5F */
Packit 08bd4c
		/* `abcdefghijklmno */
Packit 08bd4c
		1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 60 - 6F */
Packit 08bd4c
		/* pqrstuvwxyz{|}~ */
Packit 08bd4c
		1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, /* 70 - 7F */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 80 - 8F */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 90 - 9F */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* A0 - AF */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* B0 - BF */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* C0 - CF */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* D0 - DF */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* E0 - EF */
Packit 08bd4c
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* F0 - FF */
Packit 08bd4c
	};
Packit 08bd4c
	ssize_t ll;
Packit 08bd4c
	const char *pp = p;
Packit 08bd4c
	const char * const pp_end = pp + len;
Packit 08bd4c
Packit 08bd4c
	*last_is_path = 0;
Packit 08bd4c
	/*
Packit 08bd4c
	 * Skip the path-name which is quoted.
Packit 08bd4c
	 */
Packit 08bd4c
	for (;pp < pp_end; ++pp) {
Packit 08bd4c
		if (!safe_char[*(const unsigned char *)pp]) {
Packit 08bd4c
			if (*pp != ' ' && *pp != '\t' && *pp != '\r'
Packit 08bd4c
			    && *pp != '\n')
Packit 08bd4c
				f = 0;
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
		f = 1;
Packit 08bd4c
	}
Packit 08bd4c
	ll = pp_end - pp;
Packit 08bd4c
Packit 08bd4c
	/* If a path-name was not found at the first, try to check
Packit 08bd4c
	 * a mtree format(a.k.a form D) ``NetBSD's mtree -D'' creates,
Packit 08bd4c
	 * which places the path-name at the last. */
Packit 08bd4c
	if (f == 0) {
Packit 08bd4c
		const char *pb = p + len - nl;
Packit 08bd4c
		int name_len = 0;
Packit 08bd4c
		int slash;
Packit 08bd4c
Packit 08bd4c
		/* The form D accepts only a single line for an entry. */
Packit 08bd4c
		if (pb-2 >= p &&
Packit 08bd4c
		    pb[-1] == '\\' && (pb[-2] == ' ' || pb[-2] == '\t'))
Packit 08bd4c
			return (-1);
Packit 08bd4c
		if (pb-1 >= p && pb[-1] == '\\')
Packit 08bd4c
			return (-1);
Packit 08bd4c
Packit 08bd4c
		slash = 0;
Packit 08bd4c
		while (p <= --pb && *pb != ' ' && *pb != '\t') {
Packit 08bd4c
			if (!safe_char[*(const unsigned char *)pb])
Packit 08bd4c
				return (-1);
Packit 08bd4c
			name_len++;
Packit 08bd4c
			/* The pathname should have a slash in this
Packit 08bd4c
			 * format. */
Packit 08bd4c
			if (*pb == '/')
Packit 08bd4c
				slash = 1;
Packit 08bd4c
		}
Packit 08bd4c
		if (name_len == 0 || slash == 0)
Packit 08bd4c
			return (-1);
Packit 08bd4c
		/* If '/' is placed at the first in this field, this is not
Packit 08bd4c
		 * a valid filename. */
Packit 08bd4c
		if (pb[1] == '/')
Packit 08bd4c
			return (-1);
Packit 08bd4c
		ll = len - nl - name_len;
Packit 08bd4c
		pp = p;
Packit 08bd4c
		*last_is_path = 1;
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	return (bid_keyword_list(pp, ll, 0, *last_is_path));
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
#define MAX_BID_ENTRY	3
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
mtree_bid(struct archive_read *a, int best_bid)
Packit 08bd4c
{
Packit 08bd4c
	const char *signature = "#mtree";
Packit 08bd4c
	const char *p;
Packit 08bd4c
Packit 08bd4c
	(void)best_bid; /* UNUSED */
Packit 08bd4c
Packit 08bd4c
	/* Now let's look at the actual header and see if it matches. */
Packit 08bd4c
	p = __archive_read_ahead(a, strlen(signature), NULL);
Packit 08bd4c
	if (p == NULL)
Packit 08bd4c
		return (-1);
Packit 08bd4c
Packit 08bd4c
	if (memcmp(p, signature, strlen(signature)) == 0)
Packit 08bd4c
		return (8 * (int)strlen(signature));
Packit 08bd4c
Packit 08bd4c
	/*
Packit 08bd4c
	 * There is not a mtree signature. Let's try to detect mtree format.
Packit 08bd4c
	 */
Packit 08bd4c
	return (detect_form(a, NULL));
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
detect_form(struct archive_read *a, int *is_form_d)
Packit 08bd4c
{
Packit 08bd4c
	const char *p;
Packit 08bd4c
	ssize_t avail, ravail;
Packit 08bd4c
	ssize_t detected_bytes = 0, len, nl;
Packit 08bd4c
	int entry_cnt = 0, multiline = 0;
Packit 08bd4c
	int form_D = 0;/* The archive is generated by `NetBSD mtree -D'
Packit 08bd4c
			* (In this source we call it `form D') . */
Packit 08bd4c
Packit 08bd4c
	if (is_form_d != NULL)
Packit 08bd4c
		*is_form_d = 0;
Packit 08bd4c
	p = __archive_read_ahead(a, 1, &avail);
Packit 08bd4c
	if (p == NULL)
Packit 08bd4c
		return (-1);
Packit 08bd4c
	ravail = avail;
Packit 08bd4c
	for (;;) {
Packit 08bd4c
		len = next_line(a, &p, &avail, &ravail, &nl);
Packit 08bd4c
		/* The terminal character of the line should be
Packit 08bd4c
		 * a new line character, '\r\n' or '\n'. */
Packit 08bd4c
		if (len <= 0 || nl == 0)
Packit 08bd4c
			break;
Packit 08bd4c
		if (!multiline) {
Packit 08bd4c
			/* Leading whitespace is never significant,
Packit 08bd4c
			 * ignore it. */
Packit 08bd4c
			while (len > 0 && (*p == ' ' || *p == '\t')) {
Packit 08bd4c
				++p;
Packit 08bd4c
				--avail;
Packit 08bd4c
				--len;
Packit 08bd4c
			}
Packit 08bd4c
			/* Skip comment or empty line. */ 
Packit 08bd4c
			if (p[0] == '#' || p[0] == '\n' || p[0] == '\r') {
Packit 08bd4c
				p += len;
Packit 08bd4c
				avail -= len;
Packit 08bd4c
				continue;
Packit 08bd4c
			}
Packit 08bd4c
		} else {
Packit 08bd4c
			/* A continuance line; the terminal
Packit 08bd4c
			 * character of previous line was '\' character. */
Packit 08bd4c
			if (bid_keyword_list(p, len, 0, 0) <= 0)
Packit 08bd4c
				break;
Packit 08bd4c
			if (multiline == 1)
Packit 08bd4c
				detected_bytes += len;
Packit 08bd4c
			if (p[len-nl-1] != '\\') {
Packit 08bd4c
				if (multiline == 1 &&
Packit 08bd4c
				    ++entry_cnt >= MAX_BID_ENTRY)
Packit 08bd4c
					break;
Packit 08bd4c
				multiline = 0;
Packit 08bd4c
			}
Packit 08bd4c
			p += len;
Packit 08bd4c
			avail -= len;
Packit 08bd4c
			continue;
Packit 08bd4c
		}
Packit 08bd4c
		if (p[0] != '/') {
Packit 08bd4c
			int last_is_path, keywords;
Packit 08bd4c
Packit 08bd4c
			keywords = bid_entry(p, len, nl, &last_is_path);
Packit 08bd4c
			if (keywords >= 0) {
Packit 08bd4c
				detected_bytes += len;
Packit 08bd4c
				if (form_D == 0) {
Packit 08bd4c
					if (last_is_path)
Packit 08bd4c
						form_D = 1;
Packit 08bd4c
					else if (keywords > 0)
Packit 08bd4c
						/* This line is not `form D'. */
Packit 08bd4c
						form_D = -1;
Packit 08bd4c
				} else if (form_D == 1) {
Packit 08bd4c
					if (!last_is_path && keywords > 0)
Packit 08bd4c
						/* This this is not `form D'
Packit 08bd4c
						 * and We cannot accept mixed
Packit 08bd4c
						 * format. */
Packit 08bd4c
						break;
Packit 08bd4c
				}
Packit 08bd4c
				if (!last_is_path && p[len-nl-1] == '\\')
Packit 08bd4c
					/* This line continues. */
Packit 08bd4c
					multiline = 1;
Packit 08bd4c
				else {
Packit 08bd4c
					/* We've got plenty of correct lines
Packit 08bd4c
					 * to assume that this file is a mtree
Packit 08bd4c
					 * format. */
Packit 08bd4c
					if (++entry_cnt >= MAX_BID_ENTRY)
Packit 08bd4c
						break;
Packit 08bd4c
				}
Packit 08bd4c
			} else
Packit 08bd4c
				break;
Packit 08bd4c
		} else if (len > 4 && strncmp(p, "/set", 4) == 0) {
Packit 08bd4c
			if (bid_keyword_list(p+4, len-4, 0, 0) <= 0)
Packit 08bd4c
				break;
Packit 08bd4c
			/* This line continues. */
Packit 08bd4c
			if (p[len-nl-1] == '\\')
Packit 08bd4c
				multiline = 2;
Packit 08bd4c
		} else if (len > 6 && strncmp(p, "/unset", 6) == 0) {
Packit 08bd4c
			if (bid_keyword_list(p+6, len-6, 1, 0) <= 0)
Packit 08bd4c
				break;
Packit 08bd4c
			/* This line continues. */
Packit 08bd4c
			if (p[len-nl-1] == '\\')
Packit 08bd4c
				multiline = 2;
Packit 08bd4c
		} else
Packit 08bd4c
			break;
Packit 08bd4c
Packit 08bd4c
		/* Test next line. */
Packit 08bd4c
		p += len;
Packit 08bd4c
		avail -= len;
Packit 08bd4c
	}
Packit 08bd4c
	if (entry_cnt >= MAX_BID_ENTRY || (entry_cnt > 0 && len == 0)) {
Packit 08bd4c
		if (is_form_d != NULL) {
Packit 08bd4c
			if (form_D == 1)
Packit 08bd4c
				*is_form_d = 1;
Packit 08bd4c
		}
Packit 08bd4c
		return (32);
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	return (0);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * The extended mtree format permits multiple lines specifying
Packit 08bd4c
 * attributes for each file.  For those entries, only the last line
Packit 08bd4c
 * is actually used.  Practically speaking, that means we have
Packit 08bd4c
 * to read the entire mtree file into memory up front.
Packit 08bd4c
 *
Packit 08bd4c
 * The parsing is done in two steps.  First, it is decided if a line
Packit 08bd4c
 * changes the global defaults and if it is, processed accordingly.
Packit 08bd4c
 * Otherwise, the options of the line are merged with the current
Packit 08bd4c
 * global options.
Packit 08bd4c
 */
Packit 08bd4c
static int
Packit 08bd4c
add_option(struct archive_read *a, struct mtree_option **global,
Packit 08bd4c
    const char *value, size_t len)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree_option *opt;
Packit 08bd4c
Packit 08bd4c
	if ((opt = malloc(sizeof(*opt))) == NULL) {
Packit 08bd4c
		archive_set_error(&a->archive, errno, "Can't allocate memory");
Packit 08bd4c
		return (ARCHIVE_FATAL);
Packit 08bd4c
	}
Packit 08bd4c
	if ((opt->value = malloc(len + 1)) == NULL) {
Packit 08bd4c
		free(opt);
Packit 08bd4c
		archive_set_error(&a->archive, errno, "Can't allocate memory");
Packit 08bd4c
		return (ARCHIVE_FATAL);
Packit 08bd4c
	}
Packit 08bd4c
	memcpy(opt->value, value, len);
Packit 08bd4c
	opt->value[len] = '\0';
Packit 08bd4c
	opt->next = *global;
Packit 08bd4c
	*global = opt;
Packit 08bd4c
	return (ARCHIVE_OK);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static void
Packit 08bd4c
remove_option(struct mtree_option **global, const char *value, size_t len)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree_option *iter, *last;
Packit 08bd4c
Packit 08bd4c
	last = NULL;
Packit 08bd4c
	for (iter = *global; iter != NULL; last = iter, iter = iter->next) {
Packit 08bd4c
		if (strncmp(iter->value, value, len) == 0 &&
Packit 08bd4c
		    (iter->value[len] == '\0' ||
Packit 08bd4c
		     iter->value[len] == '='))
Packit 08bd4c
			break;
Packit 08bd4c
	}
Packit 08bd4c
	if (iter == NULL)
Packit 08bd4c
		return;
Packit 08bd4c
	if (last == NULL)
Packit 08bd4c
		*global = iter->next;
Packit 08bd4c
	else
Packit 08bd4c
		last->next = iter->next;
Packit 08bd4c
Packit 08bd4c
	free(iter->value);
Packit 08bd4c
	free(iter);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
process_global_set(struct archive_read *a,
Packit 08bd4c
    struct mtree_option **global, const char *line)
Packit 08bd4c
{
Packit 08bd4c
	const char *next, *eq;
Packit 08bd4c
	size_t len;
Packit 08bd4c
	int r;
Packit 08bd4c
Packit 08bd4c
	line += 4;
Packit 08bd4c
	for (;;) {
Packit 08bd4c
		next = line + strspn(line, " \t\r\n");
Packit 08bd4c
		if (*next == '\0')
Packit 08bd4c
			return (ARCHIVE_OK);
Packit 08bd4c
		line = next;
Packit 08bd4c
		next = line + strcspn(line, " \t\r\n");
Packit 08bd4c
		eq = strchr(line, '=');
Packit 08bd4c
		if (eq > next)
Packit 08bd4c
			len = next - line;
Packit 08bd4c
		else
Packit 08bd4c
			len = eq - line;
Packit 08bd4c
Packit 08bd4c
		remove_option(global, line, len);
Packit 08bd4c
		r = add_option(a, global, line, next - line);
Packit 08bd4c
		if (r != ARCHIVE_OK)
Packit 08bd4c
			return (r);
Packit 08bd4c
		line = next;
Packit 08bd4c
	}
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
process_global_unset(struct archive_read *a,
Packit 08bd4c
    struct mtree_option **global, const char *line)
Packit 08bd4c
{
Packit 08bd4c
	const char *next;
Packit 08bd4c
	size_t len;
Packit 08bd4c
Packit 08bd4c
	line += 6;
Packit 08bd4c
	if (strchr(line, '=') != NULL) {
Packit 08bd4c
		archive_set_error(&a->archive, ARCHIVE_ERRNO_MISC,
Packit 08bd4c
		    "/unset shall not contain `='");
Packit 08bd4c
		return ARCHIVE_FATAL;
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	for (;;) {
Packit 08bd4c
		next = line + strspn(line, " \t\r\n");
Packit 08bd4c
		if (*next == '\0')
Packit 08bd4c
			return (ARCHIVE_OK);
Packit 08bd4c
		line = next;
Packit 08bd4c
		len = strcspn(line, " \t\r\n");
Packit 08bd4c
Packit 08bd4c
		if (len == 3 && strncmp(line, "all", 3) == 0) {
Packit 08bd4c
			free_options(*global);
Packit 08bd4c
			*global = NULL;
Packit 08bd4c
		} else {
Packit 08bd4c
			remove_option(global, line, len);
Packit 08bd4c
		}
Packit 08bd4c
Packit 08bd4c
		line += len;
Packit 08bd4c
	}
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
process_add_entry(struct archive_read *a, struct mtree *mtree,
Packit 08bd4c
    struct mtree_option **global, const char *line, ssize_t line_len,
Packit 08bd4c
    struct mtree_entry **last_entry, int is_form_d)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree_entry *entry, *ht_iter;
Packit 08bd4c
	struct mtree_option *iter;
Packit 08bd4c
	const char *next, *eq, *name, *end;
Packit 08bd4c
	size_t name_len, len;
Packit 08bd4c
	int r, i;
Packit 08bd4c
	unsigned int ht_idx;
Packit 08bd4c
Packit 08bd4c
	if ((entry = malloc(sizeof(*entry))) == NULL) {
Packit 08bd4c
		archive_set_error(&a->archive, errno, "Can't allocate memory");
Packit 08bd4c
		return (ARCHIVE_FATAL);
Packit 08bd4c
	}
Packit 08bd4c
	entry->next = NULL;
Packit 08bd4c
	entry->options = NULL;
Packit 08bd4c
	entry->name = NULL;
Packit 08bd4c
	entry->used = 0;
Packit 08bd4c
	entry->full = 0;
Packit 08bd4c
	entry->name_hash = 0;
Packit 08bd4c
	entry->hashtable_next = NULL;
Packit 08bd4c
Packit 08bd4c
	/* Add this entry to list. */
Packit 08bd4c
	if (*last_entry == NULL)
Packit 08bd4c
		mtree->entries = entry;
Packit 08bd4c
	else
Packit 08bd4c
		(*last_entry)->next = entry;
Packit 08bd4c
	*last_entry = entry;
Packit 08bd4c
Packit 08bd4c
	if (is_form_d) {
Packit 08bd4c
		/* Filename is last item on line. */
Packit 08bd4c
		/* Adjust line_len to trim trailing whitespace */
Packit 08bd4c
		while (line_len > 0) {
Packit 08bd4c
			char last_character = line[line_len - 1];
Packit 08bd4c
			if (last_character == '\r'
Packit 08bd4c
			    || last_character == '\n'
Packit 08bd4c
			    || last_character == '\t'
Packit 08bd4c
			    || last_character == ' ') {
Packit 08bd4c
				line_len--;
Packit 08bd4c
			} else {
Packit 08bd4c
				break;
Packit 08bd4c
			}
Packit 08bd4c
		}
Packit 08bd4c
		/* Name starts after the last whitespace separator */
Packit 08bd4c
		name = line;
Packit 08bd4c
		for (i = 0; i < line_len; i++) {
Packit 08bd4c
			if (line[i] == '\r'
Packit 08bd4c
			    || line[i] == '\n'
Packit 08bd4c
			    || line[i] == '\t'
Packit 08bd4c
			    || line[i] == ' ') {
Packit 08bd4c
				name = line + i + 1;
Packit 08bd4c
			}
Packit 08bd4c
		}
Packit 08bd4c
		name_len = line + line_len - name;
Packit 08bd4c
		end = name;
Packit 08bd4c
	} else {
Packit 08bd4c
		/* Filename is first item on line */
Packit 08bd4c
		name_len = strcspn(line, " \t\r\n");
Packit 08bd4c
		name = line;
Packit 08bd4c
		line += name_len;
Packit 08bd4c
		end = line + line_len;
Packit 08bd4c
	}
Packit 08bd4c
	/* name/name_len is the name within the line. */
Packit 08bd4c
	/* line..end brackets the entire line except the name */
Packit 08bd4c
Packit 08bd4c
	if ((entry->name = malloc(name_len + 1)) == NULL) {
Packit 08bd4c
		archive_set_error(&a->archive, errno, "Can't allocate memory");
Packit 08bd4c
		return (ARCHIVE_FATAL);
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	memcpy(entry->name, name, name_len);
Packit 08bd4c
	entry->name[name_len] = '\0';
Packit 08bd4c
	parse_escapes(entry->name, entry);
Packit 08bd4c
	entry->name_hash = hash(entry->name);
Packit 08bd4c
Packit 08bd4c
	ht_idx = entry->name_hash % MTREE_HASHTABLE_SIZE;
Packit 08bd4c
	if ((ht_iter = mtree->entry_hashtable[ht_idx]) != NULL) {
Packit 08bd4c
		while (ht_iter->hashtable_next)
Packit 08bd4c
			ht_iter = ht_iter->hashtable_next;
Packit 08bd4c
		ht_iter->hashtable_next = entry;
Packit 08bd4c
	} else {
Packit 08bd4c
		mtree->entry_hashtable[ht_idx] = entry;
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	for (iter = *global; iter != NULL; iter = iter->next) {
Packit 08bd4c
		r = add_option(a, &entry->options, iter->value,
Packit 08bd4c
		    strlen(iter->value));
Packit 08bd4c
		if (r != ARCHIVE_OK)
Packit 08bd4c
			return (r);
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	for (;;) {
Packit 08bd4c
		next = line + strspn(line, " \t\r\n");
Packit 08bd4c
		if (*next == '\0')
Packit 08bd4c
			return (ARCHIVE_OK);
Packit 08bd4c
		if (next >= end)
Packit 08bd4c
			return (ARCHIVE_OK);
Packit 08bd4c
		line = next;
Packit 08bd4c
		next = line + strcspn(line, " \t\r\n");
Packit 08bd4c
		eq = strchr(line, '=');
Packit 08bd4c
		if (eq == NULL || eq > next)
Packit 08bd4c
			len = next - line;
Packit 08bd4c
		else
Packit 08bd4c
			len = eq - line;
Packit 08bd4c
Packit 08bd4c
		remove_option(&entry->options, line, len);
Packit 08bd4c
		r = add_option(a, &entry->options, line, next - line);
Packit 08bd4c
		if (r != ARCHIVE_OK)
Packit 08bd4c
			return (r);
Packit 08bd4c
		line = next;
Packit 08bd4c
	}
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
read_mtree(struct archive_read *a, struct mtree *mtree)
Packit 08bd4c
{
Packit 08bd4c
	ssize_t len;
Packit 08bd4c
	uintmax_t counter;
Packit 08bd4c
	char *p;
Packit 08bd4c
	struct mtree_option *global;
Packit 08bd4c
	struct mtree_entry *last_entry;
Packit 08bd4c
	int r, is_form_d;
Packit 08bd4c
Packit 08bd4c
	mtree->archive_format = ARCHIVE_FORMAT_MTREE;
Packit 08bd4c
	mtree->archive_format_name = "mtree";
Packit 08bd4c
Packit 08bd4c
	global = NULL;
Packit 08bd4c
	last_entry = NULL;
Packit 08bd4c
Packit 08bd4c
	(void)detect_form(a, &is_form_d);
Packit 08bd4c
Packit 08bd4c
	for (counter = 1; ; ++counter) {
Packit 08bd4c
		len = readline(a, mtree, &p, 65536);
Packit 08bd4c
		if (len == 0) {
Packit 08bd4c
			mtree->this_entry = mtree->entries;
Packit 08bd4c
			free_options(global);
Packit 08bd4c
			return (ARCHIVE_OK);
Packit 08bd4c
		}
Packit 08bd4c
		if (len < 0) {
Packit 08bd4c
			free_options(global);
Packit 08bd4c
			return ((int)len);
Packit 08bd4c
		}
Packit 08bd4c
		/* Leading whitespace is never significant, ignore it. */
Packit 08bd4c
		while (*p == ' ' || *p == '\t') {
Packit 08bd4c
			++p;
Packit 08bd4c
			--len;
Packit 08bd4c
		}
Packit 08bd4c
		/* Skip content lines and blank lines. */
Packit 08bd4c
		if (*p == '#')
Packit 08bd4c
			continue;
Packit 08bd4c
		if (*p == '\r' || *p == '\n' || *p == '\0')
Packit 08bd4c
			continue;
Packit 08bd4c
		if (*p != '/') {
Packit 08bd4c
			r = process_add_entry(a, mtree, &global, p, len,
Packit 08bd4c
			    &last_entry, is_form_d);
Packit 08bd4c
		} else if (len > 4 && strncmp(p, "/set", 4) == 0) {
Packit 08bd4c
			if (p[4] != ' ' && p[4] != '\t')
Packit 08bd4c
				break;
Packit 08bd4c
			r = process_global_set(a, &global, p);
Packit 08bd4c
		} else if (len > 6 && strncmp(p, "/unset", 6) == 0) {
Packit 08bd4c
			if (p[6] != ' ' && p[6] != '\t')
Packit 08bd4c
				break;
Packit 08bd4c
			r = process_global_unset(a, &global, p);
Packit 08bd4c
		} else
Packit 08bd4c
			break;
Packit 08bd4c
Packit 08bd4c
		if (r != ARCHIVE_OK) {
Packit 08bd4c
			free_options(global);
Packit 08bd4c
			return r;
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	archive_set_error(&a->archive, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
	    "Can't parse line %ju", counter);
Packit 08bd4c
	free_options(global);
Packit 08bd4c
	return (ARCHIVE_FATAL);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Read in the entire mtree file into memory on the first request.
Packit 08bd4c
 * Then use the next unused file to satisfy each header request.
Packit 08bd4c
 */
Packit 08bd4c
static int
Packit 08bd4c
read_header(struct archive_read *a, struct archive_entry *entry)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree *mtree;
Packit 08bd4c
	char *p;
Packit 08bd4c
	int r, use_next;
Packit 08bd4c
Packit 08bd4c
	mtree = (struct mtree *)(a->format->data);
Packit 08bd4c
Packit 08bd4c
	if (mtree->fd >= 0) {
Packit 08bd4c
		close(mtree->fd);
Packit 08bd4c
		mtree->fd = -1;
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	if (mtree->entries == NULL) {
Packit 08bd4c
		mtree->resolver = archive_entry_linkresolver_new();
Packit 08bd4c
		if (mtree->resolver == NULL)
Packit 08bd4c
			return ARCHIVE_FATAL;
Packit 08bd4c
		archive_entry_linkresolver_set_strategy(mtree->resolver,
Packit 08bd4c
		    ARCHIVE_FORMAT_MTREE);
Packit 08bd4c
		r = read_mtree(a, mtree);
Packit 08bd4c
		if (r != ARCHIVE_OK)
Packit 08bd4c
			return (r);
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	a->archive.archive_format = mtree->archive_format;
Packit 08bd4c
	a->archive.archive_format_name = mtree->archive_format_name;
Packit 08bd4c
Packit 08bd4c
	for (;;) {
Packit 08bd4c
		if (mtree->this_entry == NULL)
Packit 08bd4c
			return (ARCHIVE_EOF);
Packit 08bd4c
		if (strcmp(mtree->this_entry->name, "..") == 0) {
Packit 08bd4c
			mtree->this_entry->used = 1;
Packit 08bd4c
			if (archive_strlen(&mtree->current_dir) > 0) {
Packit 08bd4c
				/* Roll back current path. */
Packit 08bd4c
				p = mtree->current_dir.s
Packit 08bd4c
				    + mtree->current_dir.length - 1;
Packit 08bd4c
				while (p >= mtree->current_dir.s && *p != '/')
Packit 08bd4c
					--p;
Packit 08bd4c
				if (p >= mtree->current_dir.s)
Packit 08bd4c
					--p;
Packit 08bd4c
				mtree->current_dir.length
Packit 08bd4c
				    = p - mtree->current_dir.s + 1;
Packit 08bd4c
			}
Packit 08bd4c
		}
Packit 08bd4c
		if (!mtree->this_entry->used) {
Packit 08bd4c
			use_next = 0;
Packit 08bd4c
			r = parse_file(a, entry, mtree, mtree->this_entry,
Packit 08bd4c
				&use_next);
Packit 08bd4c
			if (use_next == 0)
Packit 08bd4c
				return (r);
Packit 08bd4c
		}
Packit 08bd4c
		mtree->this_entry = mtree->this_entry->next;
Packit 08bd4c
	}
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * A single file can have multiple lines contribute specifications.
Packit 08bd4c
 * Parse as many lines as necessary, then pull additional information
Packit 08bd4c
 * from a backing file on disk as necessary.
Packit 08bd4c
 */
Packit 08bd4c
static int
Packit 08bd4c
parse_file(struct archive_read *a, struct archive_entry *entry,
Packit 08bd4c
    struct mtree *mtree, struct mtree_entry *mentry, int *use_next)
Packit 08bd4c
{
Packit 08bd4c
	const char *path;
Packit 08bd4c
	struct stat st_storage, *st;
Packit 08bd4c
	struct mtree_entry *mp;
Packit 08bd4c
	struct archive_entry *sparse_entry;
Packit 08bd4c
	int r = ARCHIVE_OK, r1, parsed_kws;
Packit 08bd4c
Packit 08bd4c
	mentry->used = 1;
Packit 08bd4c
Packit 08bd4c
	/* Initialize reasonable defaults. */
Packit 08bd4c
	archive_entry_set_filetype(entry, AE_IFREG);
Packit 08bd4c
	archive_entry_set_size(entry, 0);
Packit 08bd4c
	archive_string_empty(&mtree->contents_name);
Packit 08bd4c
Packit 08bd4c
	/* Parse options from this line. */
Packit 08bd4c
	parsed_kws = 0;
Packit 08bd4c
	r = parse_line(a, entry, mtree, mentry, &parsed_kws);
Packit 08bd4c
Packit 08bd4c
	if (mentry->full) {
Packit 08bd4c
		archive_entry_copy_pathname(entry, mentry->name);
Packit 08bd4c
		/*
Packit 08bd4c
		 * "Full" entries are allowed to have multiple lines
Packit 08bd4c
		 * and those lines aren't required to be adjacent.  We
Packit 08bd4c
		 * don't support multiple lines for "relative" entries
Packit 08bd4c
		 * nor do we make any attempt to merge data from
Packit 08bd4c
		 * separate "relative" and "full" entries.  (Merging
Packit 08bd4c
		 * "relative" and "full" entries would require dealing
Packit 08bd4c
		 * with pathname canonicalization, which is a very
Packit 08bd4c
		 * tricky subject.)
Packit 08bd4c
		 */
Packit 08bd4c
		for (mp = mentry->hashtable_next; mp != NULL; mp = mp->hashtable_next) {
Packit 08bd4c
			if (mp->full && !mp->used
Packit 08bd4c
					&& mentry->name_hash == mp->name_hash
Packit 08bd4c
					&& strcmp(mentry->name, mp->name) == 0) {
Packit 08bd4c
				/* Later lines override earlier ones. */
Packit 08bd4c
				mp->used = 1;
Packit 08bd4c
				r1 = parse_line(a, entry, mtree, mp,
Packit 08bd4c
				    &parsed_kws);
Packit 08bd4c
				if (r1 < r)
Packit 08bd4c
					r = r1;
Packit 08bd4c
			}
Packit 08bd4c
		}
Packit 08bd4c
	} else {
Packit 08bd4c
		/*
Packit 08bd4c
		 * Relative entries require us to construct
Packit 08bd4c
		 * the full path and possibly update the
Packit 08bd4c
		 * current directory.
Packit 08bd4c
		 */
Packit 08bd4c
		size_t n = archive_strlen(&mtree->current_dir);
Packit 08bd4c
		if (n > 0)
Packit 08bd4c
			archive_strcat(&mtree->current_dir, "/");
Packit 08bd4c
		archive_strcat(&mtree->current_dir, mentry->name);
Packit 08bd4c
		archive_entry_copy_pathname(entry, mtree->current_dir.s);
Packit 08bd4c
		if (archive_entry_filetype(entry) != AE_IFDIR)
Packit 08bd4c
			mtree->current_dir.length = n;
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	if (mtree->checkfs) {
Packit 08bd4c
		/*
Packit 08bd4c
		 * Try to open and stat the file to get the real size
Packit 08bd4c
		 * and other file info.  It would be nice to avoid
Packit 08bd4c
		 * this here so that getting a listing of an mtree
Packit 08bd4c
		 * wouldn't require opening every referenced contents
Packit 08bd4c
		 * file.  But then we wouldn't know the actual
Packit 08bd4c
		 * contents size, so I don't see a really viable way
Packit 08bd4c
		 * around this.  (Also, we may want to someday pull
Packit 08bd4c
		 * other unspecified info from the contents file on
Packit 08bd4c
		 * disk.)
Packit 08bd4c
		 */
Packit 08bd4c
		mtree->fd = -1;
Packit 08bd4c
		if (archive_strlen(&mtree->contents_name) > 0)
Packit 08bd4c
			path = mtree->contents_name.s;
Packit 08bd4c
		else
Packit 08bd4c
			path = archive_entry_pathname(entry);
Packit 08bd4c
Packit 08bd4c
		if (archive_entry_filetype(entry) == AE_IFREG ||
Packit 08bd4c
				archive_entry_filetype(entry) == AE_IFDIR) {
Packit 08bd4c
			mtree->fd = open(path, O_RDONLY | O_BINARY | O_CLOEXEC);
Packit 08bd4c
			__archive_ensure_cloexec_flag(mtree->fd);
Packit 08bd4c
			if (mtree->fd == -1 &&
Packit 08bd4c
				(errno != ENOENT ||
Packit 08bd4c
				 archive_strlen(&mtree->contents_name) > 0)) {
Packit 08bd4c
				archive_set_error(&a->archive, errno,
Packit 08bd4c
						"Can't open %s", path);
Packit 08bd4c
				r = ARCHIVE_WARN;
Packit 08bd4c
			}
Packit 08bd4c
		}
Packit 08bd4c
Packit 08bd4c
		st = &st_storage;
Packit 08bd4c
		if (mtree->fd >= 0) {
Packit 08bd4c
			if (fstat(mtree->fd, st) == -1) {
Packit 08bd4c
				archive_set_error(&a->archive, errno,
Packit 08bd4c
						"Could not fstat %s", path);
Packit 08bd4c
				r = ARCHIVE_WARN;
Packit 08bd4c
				/* If we can't stat it, don't keep it open. */
Packit 08bd4c
				close(mtree->fd);
Packit 08bd4c
				mtree->fd = -1;
Packit 08bd4c
				st = NULL;
Packit 08bd4c
			}
Packit 08bd4c
		} else if (lstat(path, st) == -1) {
Packit 08bd4c
			st = NULL;
Packit 08bd4c
		}
Packit 08bd4c
Packit 08bd4c
		/*
Packit 08bd4c
		 * Check for a mismatch between the type in the specification
Packit 08bd4c
		 * and the type of the contents object on disk.
Packit 08bd4c
		 */
Packit 08bd4c
		if (st != NULL) {
Packit 08bd4c
			if (((st->st_mode & S_IFMT) == S_IFREG &&
Packit 08bd4c
			      archive_entry_filetype(entry) == AE_IFREG)
Packit 08bd4c
#ifdef S_IFLNK
Packit 08bd4c
			  ||((st->st_mode & S_IFMT) == S_IFLNK &&
Packit 08bd4c
			      archive_entry_filetype(entry) == AE_IFLNK)
Packit 08bd4c
#endif
Packit 08bd4c
#ifdef S_IFSOCK
Packit 08bd4c
			  ||((st->st_mode & S_IFSOCK) == S_IFSOCK &&
Packit 08bd4c
			      archive_entry_filetype(entry) == AE_IFSOCK)
Packit 08bd4c
#endif
Packit 08bd4c
#ifdef S_IFCHR
Packit 08bd4c
			  ||((st->st_mode & S_IFMT) == S_IFCHR &&
Packit 08bd4c
			      archive_entry_filetype(entry) == AE_IFCHR)
Packit 08bd4c
#endif
Packit 08bd4c
#ifdef S_IFBLK
Packit 08bd4c
			  ||((st->st_mode & S_IFMT) == S_IFBLK &&
Packit 08bd4c
			      archive_entry_filetype(entry) == AE_IFBLK)
Packit 08bd4c
#endif
Packit 08bd4c
			  ||((st->st_mode & S_IFMT) == S_IFDIR &&
Packit 08bd4c
			      archive_entry_filetype(entry) == AE_IFDIR)
Packit 08bd4c
#ifdef S_IFIFO
Packit 08bd4c
			  ||((st->st_mode & S_IFMT) == S_IFIFO &&
Packit 08bd4c
			      archive_entry_filetype(entry) == AE_IFIFO)
Packit 08bd4c
#endif
Packit 08bd4c
			) {
Packit 08bd4c
				/* Types match. */
Packit 08bd4c
			} else {
Packit 08bd4c
				/* Types don't match; bail out gracefully. */
Packit 08bd4c
				if (mtree->fd >= 0)
Packit 08bd4c
					close(mtree->fd);
Packit 08bd4c
				mtree->fd = -1;
Packit 08bd4c
				if (parsed_kws & MTREE_HAS_OPTIONAL) {
Packit 08bd4c
					/* It's not an error for an optional
Packit 08bd4c
					 * entry to not match disk. */
Packit 08bd4c
					*use_next = 1;
Packit 08bd4c
				} else if (r == ARCHIVE_OK) {
Packit 08bd4c
					archive_set_error(&a->archive,
Packit 08bd4c
					    ARCHIVE_ERRNO_MISC,
Packit 08bd4c
					    "mtree specification has different"
Packit 08bd4c
					    " type for %s",
Packit 08bd4c
					    archive_entry_pathname(entry));
Packit 08bd4c
					r = ARCHIVE_WARN;
Packit 08bd4c
				}
Packit 08bd4c
				return (r);
Packit 08bd4c
			}
Packit 08bd4c
		}
Packit 08bd4c
Packit 08bd4c
		/*
Packit 08bd4c
		 * If there is a contents file on disk, pick some of the
Packit 08bd4c
		 * metadata from that file.  For most of these, we only
Packit 08bd4c
		 * set it from the contents if it wasn't already parsed
Packit 08bd4c
		 * from the specification.
Packit 08bd4c
		 */
Packit 08bd4c
		if (st != NULL) {
Packit 08bd4c
			if (((parsed_kws & MTREE_HAS_DEVICE) == 0 ||
Packit 08bd4c
				(parsed_kws & MTREE_HAS_NOCHANGE) != 0) &&
Packit 08bd4c
				(archive_entry_filetype(entry) == AE_IFCHR ||
Packit 08bd4c
				 archive_entry_filetype(entry) == AE_IFBLK))
Packit 08bd4c
				archive_entry_set_rdev(entry, st->st_rdev);
Packit 08bd4c
			if ((parsed_kws & (MTREE_HAS_GID | MTREE_HAS_GNAME))
Packit 08bd4c
				== 0 ||
Packit 08bd4c
			    (parsed_kws & MTREE_HAS_NOCHANGE) != 0)
Packit 08bd4c
				archive_entry_set_gid(entry, st->st_gid);
Packit 08bd4c
			if ((parsed_kws & (MTREE_HAS_UID | MTREE_HAS_UNAME))
Packit 08bd4c
				== 0 ||
Packit 08bd4c
			    (parsed_kws & MTREE_HAS_NOCHANGE) != 0)
Packit 08bd4c
				archive_entry_set_uid(entry, st->st_uid);
Packit 08bd4c
			if ((parsed_kws & MTREE_HAS_MTIME) == 0 ||
Packit 08bd4c
			    (parsed_kws & MTREE_HAS_NOCHANGE) != 0) {
Packit 08bd4c
#if HAVE_STRUCT_STAT_ST_MTIMESPEC_TV_NSEC
Packit 08bd4c
				archive_entry_set_mtime(entry, st->st_mtime,
Packit 08bd4c
						st->st_mtimespec.tv_nsec);
Packit 08bd4c
#elif HAVE_STRUCT_STAT_ST_MTIM_TV_NSEC
Packit 08bd4c
				archive_entry_set_mtime(entry, st->st_mtime,
Packit 08bd4c
						st->st_mtim.tv_nsec);
Packit 08bd4c
#elif HAVE_STRUCT_STAT_ST_MTIME_N
Packit 08bd4c
				archive_entry_set_mtime(entry, st->st_mtime,
Packit 08bd4c
						st->st_mtime_n);
Packit 08bd4c
#elif HAVE_STRUCT_STAT_ST_UMTIME
Packit 08bd4c
				archive_entry_set_mtime(entry, st->st_mtime,
Packit 08bd4c
						st->st_umtime*1000);
Packit 08bd4c
#elif HAVE_STRUCT_STAT_ST_MTIME_USEC
Packit 08bd4c
				archive_entry_set_mtime(entry, st->st_mtime,
Packit 08bd4c
						st->st_mtime_usec*1000);
Packit 08bd4c
#else
Packit 08bd4c
				archive_entry_set_mtime(entry, st->st_mtime, 0);
Packit 08bd4c
#endif
Packit 08bd4c
			}
Packit 08bd4c
			if ((parsed_kws & MTREE_HAS_NLINK) == 0 ||
Packit 08bd4c
			    (parsed_kws & MTREE_HAS_NOCHANGE) != 0)
Packit 08bd4c
				archive_entry_set_nlink(entry, st->st_nlink);
Packit 08bd4c
			if ((parsed_kws & MTREE_HAS_PERM) == 0 ||
Packit 08bd4c
			    (parsed_kws & MTREE_HAS_NOCHANGE) != 0)
Packit 08bd4c
				archive_entry_set_perm(entry, st->st_mode);
Packit 08bd4c
			if ((parsed_kws & MTREE_HAS_SIZE) == 0 ||
Packit 08bd4c
			    (parsed_kws & MTREE_HAS_NOCHANGE) != 0)
Packit 08bd4c
				archive_entry_set_size(entry, st->st_size);
Packit 08bd4c
			archive_entry_set_ino(entry, st->st_ino);
Packit 08bd4c
			archive_entry_set_dev(entry, st->st_dev);
Packit 08bd4c
Packit 08bd4c
			archive_entry_linkify(mtree->resolver, &entry,
Packit 08bd4c
				&sparse_entry);
Packit 08bd4c
		} else if (parsed_kws & MTREE_HAS_OPTIONAL) {
Packit 08bd4c
			/*
Packit 08bd4c
			 * Couldn't open the entry, stat it or the on-disk type
Packit 08bd4c
			 * didn't match.  If this entry is optional, just
Packit 08bd4c
			 * ignore it and read the next header entry.
Packit 08bd4c
			 */
Packit 08bd4c
			*use_next = 1;
Packit 08bd4c
			return ARCHIVE_OK;
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	mtree->cur_size = archive_entry_size(entry);
Packit 08bd4c
	mtree->offset = 0;
Packit 08bd4c
Packit 08bd4c
	return r;
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Each line contains a sequence of keywords.
Packit 08bd4c
 */
Packit 08bd4c
static int
Packit 08bd4c
parse_line(struct archive_read *a, struct archive_entry *entry,
Packit 08bd4c
    struct mtree *mtree, struct mtree_entry *mp, int *parsed_kws)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree_option *iter;
Packit 08bd4c
	int r = ARCHIVE_OK, r1;
Packit 08bd4c
Packit 08bd4c
	for (iter = mp->options; iter != NULL; iter = iter->next) {
Packit 08bd4c
		r1 = parse_keyword(a, mtree, entry, iter, parsed_kws);
Packit 08bd4c
		if (r1 < r)
Packit 08bd4c
			r = r1;
Packit 08bd4c
	}
Packit 08bd4c
	if (r == ARCHIVE_OK && (*parsed_kws & MTREE_HAS_TYPE) == 0) {
Packit 08bd4c
		archive_set_error(&a->archive, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
		    "Missing type keyword in mtree specification");
Packit 08bd4c
		return (ARCHIVE_WARN);
Packit 08bd4c
	}
Packit 08bd4c
	return (r);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Device entries have one of the following forms:
Packit 08bd4c
 *  - raw dev_t
Packit 08bd4c
 *  - format,major,minor[,subdevice]
Packit 08bd4c
 * When parsing succeeded, `pdev' will contain the appropriate dev_t value.
Packit 08bd4c
 */
Packit 08bd4c
Packit 08bd4c
/* strsep() is not in C90, but strcspn() is. */
Packit 08bd4c
/* Taken from http://unixpapa.com/incnote/string.html */
Packit 08bd4c
static char *
Packit 08bd4c
la_strsep(char **sp, const char *sep)
Packit 08bd4c
{
Packit 08bd4c
	char *p, *s;
Packit 08bd4c
	if (sp == NULL || *sp == NULL || **sp == '\0')
Packit 08bd4c
		return(NULL);
Packit 08bd4c
	s = *sp;
Packit 08bd4c
	p = s + strcspn(s, sep);
Packit 08bd4c
	if (*p != '\0')
Packit 08bd4c
		*p++ = '\0';
Packit 08bd4c
	*sp = p;
Packit 08bd4c
	return(s);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
parse_device(dev_t *pdev, struct archive *a, char *val)
Packit 08bd4c
{
Packit 08bd4c
#define MAX_PACK_ARGS 3
Packit 08bd4c
	unsigned long numbers[MAX_PACK_ARGS];
Packit 08bd4c
	char *p, *dev;
Packit 08bd4c
	int argc;
Packit 08bd4c
	pack_t *pack;
Packit 08bd4c
	dev_t result;
Packit 08bd4c
	const char *error = NULL;
Packit 08bd4c
Packit 08bd4c
	memset(pdev, 0, sizeof(*pdev));
Packit 08bd4c
	if ((dev = strchr(val, ',')) != NULL) {
Packit 08bd4c
		/*
Packit 08bd4c
		 * Device's major/minor are given in a specified format.
Packit 08bd4c
		 * Decode and pack it accordingly.
Packit 08bd4c
		 */
Packit 08bd4c
		*dev++ = '\0';
Packit 08bd4c
		if ((pack = pack_find(val)) == NULL) {
Packit 08bd4c
			archive_set_error(a, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
			    "Unknown format `%s'", val);
Packit 08bd4c
			return ARCHIVE_WARN;
Packit 08bd4c
		}
Packit 08bd4c
		argc = 0;
Packit 08bd4c
		while ((p = la_strsep(&dev, ",")) != NULL) {
Packit 08bd4c
			if (*p == '\0') {
Packit 08bd4c
				archive_set_error(a, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
				    "Missing number");
Packit 08bd4c
				return ARCHIVE_WARN;
Packit 08bd4c
			}
Packit 08bd4c
			if (argc >= MAX_PACK_ARGS) {
Packit 08bd4c
				archive_set_error(a, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
				    "Too many arguments");
Packit 08bd4c
				return ARCHIVE_WARN;
Packit 08bd4c
			}
Packit 08bd4c
			numbers[argc++] = (unsigned long)mtree_atol(&p, 0);
Packit 08bd4c
		}
Packit 08bd4c
		if (argc < 2) {
Packit 08bd4c
			archive_set_error(a, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
			    "Not enough arguments");
Packit 08bd4c
			return ARCHIVE_WARN;
Packit 08bd4c
		}
Packit 08bd4c
		result = (*pack)(argc, numbers, &error);
Packit 08bd4c
		if (error != NULL) {
Packit 08bd4c
			archive_set_error(a, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
			    "%s", error);
Packit 08bd4c
			return ARCHIVE_WARN;
Packit 08bd4c
		}
Packit 08bd4c
	} else {
Packit 08bd4c
		/* file system raw value. */
Packit 08bd4c
		result = (dev_t)mtree_atol(&val, 0);
Packit 08bd4c
	}
Packit 08bd4c
	*pdev = result;
Packit 08bd4c
	return ARCHIVE_OK;
Packit 08bd4c
#undef MAX_PACK_ARGS
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Parse a single keyword and its value.
Packit 08bd4c
 */
Packit 08bd4c
static int
Packit 08bd4c
parse_keyword(struct archive_read *a, struct mtree *mtree,
Packit 08bd4c
    struct archive_entry *entry, struct mtree_option *opt, int *parsed_kws)
Packit 08bd4c
{
Packit 08bd4c
	char *val, *key;
Packit 08bd4c
Packit 08bd4c
	key = opt->value;
Packit 08bd4c
Packit 08bd4c
	if (*key == '\0')
Packit 08bd4c
		return (ARCHIVE_OK);
Packit 08bd4c
Packit 08bd4c
	if (strcmp(key, "nochange") == 0) {
Packit 08bd4c
		*parsed_kws |= MTREE_HAS_NOCHANGE;
Packit 08bd4c
		return (ARCHIVE_OK);
Packit 08bd4c
	}
Packit 08bd4c
	if (strcmp(key, "optional") == 0) {
Packit 08bd4c
		*parsed_kws |= MTREE_HAS_OPTIONAL;
Packit 08bd4c
		return (ARCHIVE_OK);
Packit 08bd4c
	}
Packit 08bd4c
	if (strcmp(key, "ignore") == 0) {
Packit 08bd4c
		/*
Packit 08bd4c
		 * The mtree processing is not recursive, so
Packit 08bd4c
		 * recursion will only happen for explicitly listed
Packit 08bd4c
		 * entries.
Packit 08bd4c
		 */
Packit 08bd4c
		return (ARCHIVE_OK);
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	val = strchr(key, '=');
Packit 08bd4c
	if (val == NULL) {
Packit 08bd4c
		archive_set_error(&a->archive, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
		    "Malformed attribute \"%s\" (%d)", key, key[0]);
Packit 08bd4c
		return (ARCHIVE_WARN);
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	*val = '\0';
Packit 08bd4c
	++val;
Packit 08bd4c
Packit 08bd4c
	switch (key[0]) {
Packit 08bd4c
	case 'c':
Packit 08bd4c
		if (strcmp(key, "content") == 0
Packit 08bd4c
		    || strcmp(key, "contents") == 0) {
Packit 08bd4c
			parse_escapes(val, NULL);
Packit 08bd4c
			archive_strcpy(&mtree->contents_name, val);
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
		if (strcmp(key, "cksum") == 0)
Packit 08bd4c
			break;
Packit 08bd4c
	case 'd':
Packit 08bd4c
		if (strcmp(key, "device") == 0) {
Packit 08bd4c
			/* stat(2) st_rdev field, e.g. the major/minor IDs
Packit 08bd4c
			 * of a char/block special file */
Packit 08bd4c
			int r;
Packit 08bd4c
			dev_t dev;
Packit 08bd4c
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_DEVICE;
Packit 08bd4c
			r = parse_device(&dev, &a->archive, val);
Packit 08bd4c
			if (r == ARCHIVE_OK)
Packit 08bd4c
				archive_entry_set_rdev(entry, dev);
Packit 08bd4c
			return r;
Packit 08bd4c
		}
Packit 08bd4c
	case 'f':
Packit 08bd4c
		if (strcmp(key, "flags") == 0) {
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_FFLAGS;
Packit 08bd4c
			archive_entry_copy_fflags_text(entry, val);
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	case 'g':
Packit 08bd4c
		if (strcmp(key, "gid") == 0) {
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_GID;
Packit 08bd4c
			archive_entry_set_gid(entry, mtree_atol(&val, 10));
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
		if (strcmp(key, "gname") == 0) {
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_GNAME;
Packit 08bd4c
			archive_entry_copy_gname(entry, val);
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	case 'i':
Packit 08bd4c
		if (strcmp(key, "inode") == 0) {
Packit 08bd4c
			archive_entry_set_ino(entry, mtree_atol(&val, 10));
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	case 'l':
Packit 08bd4c
		if (strcmp(key, "link") == 0) {
Packit 08bd4c
			archive_entry_copy_symlink(entry, val);
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	case 'm':
Packit 08bd4c
		if (strcmp(key, "md5") == 0 || strcmp(key, "md5digest") == 0)
Packit 08bd4c
			break;
Packit 08bd4c
		if (strcmp(key, "mode") == 0) {
Packit 08bd4c
			if (val[0] >= '0' && val[0] <= '7') {
Packit 08bd4c
				*parsed_kws |= MTREE_HAS_PERM;
Packit 08bd4c
				archive_entry_set_perm(entry,
Packit 08bd4c
				    (mode_t)mtree_atol(&val, 8));
Packit 08bd4c
			} else {
Packit 08bd4c
				archive_set_error(&a->archive,
Packit 08bd4c
				    ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
				    "Symbolic or non-octal mode \"%s\" unsupported", val);
Packit 08bd4c
				return ARCHIVE_WARN;
Packit 08bd4c
			}
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	case 'n':
Packit 08bd4c
		if (strcmp(key, "nlink") == 0) {
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_NLINK;
Packit 08bd4c
			archive_entry_set_nlink(entry,
Packit 08bd4c
				(unsigned int)mtree_atol(&val, 10));
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	case 'r':
Packit 08bd4c
		if (strcmp(key, "resdevice") == 0) {
Packit 08bd4c
			/* stat(2) st_dev field, e.g. the device ID where the
Packit 08bd4c
			 * inode resides */
Packit 08bd4c
			int r;
Packit 08bd4c
			dev_t dev;
Packit 08bd4c
Packit 08bd4c
			r = parse_device(&dev, &a->archive, val);
Packit 08bd4c
			if (r == ARCHIVE_OK)
Packit 08bd4c
				archive_entry_set_dev(entry, dev);
Packit 08bd4c
			return r;
Packit 08bd4c
		}
Packit 08bd4c
		if (strcmp(key, "rmd160") == 0 ||
Packit 08bd4c
		    strcmp(key, "rmd160digest") == 0)
Packit 08bd4c
			break;
Packit 08bd4c
	case 's':
Packit 08bd4c
		if (strcmp(key, "sha1") == 0 || strcmp(key, "sha1digest") == 0)
Packit 08bd4c
			break;
Packit 08bd4c
		if (strcmp(key, "sha256") == 0 ||
Packit 08bd4c
		    strcmp(key, "sha256digest") == 0)
Packit 08bd4c
			break;
Packit 08bd4c
		if (strcmp(key, "sha384") == 0 ||
Packit 08bd4c
		    strcmp(key, "sha384digest") == 0)
Packit 08bd4c
			break;
Packit 08bd4c
		if (strcmp(key, "sha512") == 0 ||
Packit 08bd4c
		    strcmp(key, "sha512digest") == 0)
Packit 08bd4c
			break;
Packit 08bd4c
		if (strcmp(key, "size") == 0) {
Packit 08bd4c
			archive_entry_set_size(entry, mtree_atol(&val, 10));
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	case 't':
Packit 08bd4c
		if (strcmp(key, "tags") == 0) {
Packit 08bd4c
			/*
Packit 08bd4c
			 * Comma delimited list of tags.
Packit 08bd4c
			 * Ignore the tags for now, but the interface
Packit 08bd4c
			 * should be extended to allow inclusion/exclusion.
Packit 08bd4c
			 */
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
		if (strcmp(key, "time") == 0) {
Packit 08bd4c
			int64_t m;
Packit 08bd4c
			int64_t my_time_t_max = get_time_t_max();
Packit 08bd4c
			int64_t my_time_t_min = get_time_t_min();
Packit 08bd4c
			long ns = 0;
Packit 08bd4c
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_MTIME;
Packit 08bd4c
			m = mtree_atol(&val, 10);
Packit 08bd4c
			/* Replicate an old mtree bug:
Packit 08bd4c
			 * 123456789.1 represents 123456789
Packit 08bd4c
			 * seconds and 1 nanosecond. */
Packit 08bd4c
			if (*val == '.') {
Packit 08bd4c
				++val;
Packit 08bd4c
				ns = (long)mtree_atol(&val, 10);
Packit 08bd4c
				if (ns < 0)
Packit 08bd4c
					ns = 0;
Packit 08bd4c
				else if (ns > 999999999)
Packit 08bd4c
					ns = 999999999;
Packit 08bd4c
			}
Packit 08bd4c
			if (m > my_time_t_max)
Packit 08bd4c
				m = my_time_t_max;
Packit 08bd4c
			else if (m < my_time_t_min)
Packit 08bd4c
				m = my_time_t_min;
Packit 08bd4c
			archive_entry_set_mtime(entry, (time_t)m, ns);
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
		if (strcmp(key, "type") == 0) {
Packit 08bd4c
			switch (val[0]) {
Packit 08bd4c
			case 'b':
Packit 08bd4c
				if (strcmp(val, "block") == 0) {
Packit 08bd4c
					archive_entry_set_filetype(entry, AE_IFBLK);
Packit 08bd4c
					break;
Packit 08bd4c
				}
Packit 08bd4c
			case 'c':
Packit 08bd4c
				if (strcmp(val, "char") == 0) {
Packit 08bd4c
					archive_entry_set_filetype(entry,
Packit 08bd4c
						AE_IFCHR);
Packit 08bd4c
					break;
Packit 08bd4c
				}
Packit 08bd4c
			case 'd':
Packit 08bd4c
				if (strcmp(val, "dir") == 0) {
Packit 08bd4c
					archive_entry_set_filetype(entry,
Packit 08bd4c
						AE_IFDIR);
Packit 08bd4c
					break;
Packit 08bd4c
				}
Packit 08bd4c
			case 'f':
Packit 08bd4c
				if (strcmp(val, "fifo") == 0) {
Packit 08bd4c
					archive_entry_set_filetype(entry,
Packit 08bd4c
						AE_IFIFO);
Packit 08bd4c
					break;
Packit 08bd4c
				}
Packit 08bd4c
				if (strcmp(val, "file") == 0) {
Packit 08bd4c
					archive_entry_set_filetype(entry,
Packit 08bd4c
						AE_IFREG);
Packit 08bd4c
					break;
Packit 08bd4c
				}
Packit 08bd4c
			case 'l':
Packit 08bd4c
				if (strcmp(val, "link") == 0) {
Packit 08bd4c
					archive_entry_set_filetype(entry,
Packit 08bd4c
						AE_IFLNK);
Packit 08bd4c
					break;
Packit 08bd4c
				}
Packit 08bd4c
			default:
Packit 08bd4c
				archive_set_error(&a->archive,
Packit 08bd4c
				    ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
				    "Unrecognized file type \"%s\"; "
Packit 08bd4c
				    "assuming \"file\"", val);
Packit 08bd4c
				archive_entry_set_filetype(entry, AE_IFREG);
Packit 08bd4c
				return (ARCHIVE_WARN);
Packit 08bd4c
			}
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_TYPE;
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	case 'u':
Packit 08bd4c
		if (strcmp(key, "uid") == 0) {
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_UID;
Packit 08bd4c
			archive_entry_set_uid(entry, mtree_atol(&val, 10));
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
		if (strcmp(key, "uname") == 0) {
Packit 08bd4c
			*parsed_kws |= MTREE_HAS_UNAME;
Packit 08bd4c
			archive_entry_copy_uname(entry, val);
Packit 08bd4c
			break;
Packit 08bd4c
		}
Packit 08bd4c
	default:
Packit 08bd4c
		archive_set_error(&a->archive, ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
		    "Unrecognized key %s=%s", key, val);
Packit 08bd4c
		return (ARCHIVE_WARN);
Packit 08bd4c
	}
Packit 08bd4c
	return (ARCHIVE_OK);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static int
Packit 08bd4c
read_data(struct archive_read *a, const void **buff, size_t *size,
Packit 08bd4c
    int64_t *offset)
Packit 08bd4c
{
Packit 08bd4c
	size_t bytes_to_read;
Packit 08bd4c
	ssize_t bytes_read;
Packit 08bd4c
	struct mtree *mtree;
Packit 08bd4c
Packit 08bd4c
	mtree = (struct mtree *)(a->format->data);
Packit 08bd4c
	if (mtree->fd < 0) {
Packit 08bd4c
		*buff = NULL;
Packit 08bd4c
		*offset = 0;
Packit 08bd4c
		*size = 0;
Packit 08bd4c
		return (ARCHIVE_EOF);
Packit 08bd4c
	}
Packit 08bd4c
	if (mtree->buff == NULL) {
Packit 08bd4c
		mtree->buffsize = 64 * 1024;
Packit 08bd4c
		mtree->buff = malloc(mtree->buffsize);
Packit 08bd4c
		if (mtree->buff == NULL) {
Packit 08bd4c
			archive_set_error(&a->archive, ENOMEM,
Packit 08bd4c
			    "Can't allocate memory");
Packit 08bd4c
			return (ARCHIVE_FATAL);
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	*buff = mtree->buff;
Packit 08bd4c
	*offset = mtree->offset;
Packit 08bd4c
	if ((int64_t)mtree->buffsize > mtree->cur_size - mtree->offset)
Packit 08bd4c
		bytes_to_read = (size_t)(mtree->cur_size - mtree->offset);
Packit 08bd4c
	else
Packit 08bd4c
		bytes_to_read = mtree->buffsize;
Packit 08bd4c
	bytes_read = read(mtree->fd, mtree->buff, bytes_to_read);
Packit 08bd4c
	if (bytes_read < 0) {
Packit 08bd4c
		archive_set_error(&a->archive, errno, "Can't read");
Packit 08bd4c
		return (ARCHIVE_WARN);
Packit 08bd4c
	}
Packit 08bd4c
	if (bytes_read == 0) {
Packit 08bd4c
		*size = 0;
Packit 08bd4c
		return (ARCHIVE_EOF);
Packit 08bd4c
	}
Packit 08bd4c
	mtree->offset += bytes_read;
Packit 08bd4c
	*size = bytes_read;
Packit 08bd4c
	return (ARCHIVE_OK);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/* Skip does nothing except possibly close the contents file. */
Packit 08bd4c
static int
Packit 08bd4c
skip(struct archive_read *a)
Packit 08bd4c
{
Packit 08bd4c
	struct mtree *mtree;
Packit 08bd4c
Packit 08bd4c
	mtree = (struct mtree *)(a->format->data);
Packit 08bd4c
	if (mtree->fd >= 0) {
Packit 08bd4c
		close(mtree->fd);
Packit 08bd4c
		mtree->fd = -1;
Packit 08bd4c
	}
Packit 08bd4c
	return (ARCHIVE_OK);
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Since parsing backslash sequences always makes strings shorter,
Packit 08bd4c
 * we can always do this conversion in-place.
Packit 08bd4c
 */
Packit 08bd4c
static void
Packit 08bd4c
parse_escapes(char *src, struct mtree_entry *mentry)
Packit 08bd4c
{
Packit 08bd4c
	char *dest = src;
Packit 08bd4c
	char c;
Packit 08bd4c
Packit 08bd4c
	if (mentry != NULL && strcmp(src, ".") == 0)
Packit 08bd4c
		mentry->full = 1;
Packit 08bd4c
Packit 08bd4c
	while (*src != '\0') {
Packit 08bd4c
		c = *src++;
Packit 08bd4c
		if (c == '/' && mentry != NULL)
Packit 08bd4c
			mentry->full = 1;
Packit 08bd4c
		if (c == '\\') {
Packit 08bd4c
			switch (src[0]) {
Packit 08bd4c
			case '0':
Packit 08bd4c
				if (src[1] < '0' || src[1] > '7') {
Packit 08bd4c
					c = 0;
Packit 08bd4c
					++src;
Packit 08bd4c
					break;
Packit 08bd4c
				}
Packit 08bd4c
				/* FALLTHROUGH */
Packit 08bd4c
			case '1':
Packit 08bd4c
			case '2':
Packit 08bd4c
			case '3':
Packit 08bd4c
				if (src[1] >= '0' && src[1] <= '7' &&
Packit 08bd4c
				    src[2] >= '0' && src[2] <= '7') {
Packit 08bd4c
					c = (src[0] - '0') << 6;
Packit 08bd4c
					c |= (src[1] - '0') << 3;
Packit 08bd4c
					c |= (src[2] - '0');
Packit 08bd4c
					src += 3;
Packit 08bd4c
				}
Packit 08bd4c
				break;
Packit 08bd4c
			case 'a':
Packit 08bd4c
				c = '\a';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			case 'b':
Packit 08bd4c
				c = '\b';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			case 'f':
Packit 08bd4c
				c = '\f';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			case 'n':
Packit 08bd4c
				c = '\n';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			case 'r':
Packit 08bd4c
				c = '\r';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			case 's':
Packit 08bd4c
				c = ' ';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			case 't':
Packit 08bd4c
				c = '\t';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			case 'v':
Packit 08bd4c
				c = '\v';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			case '\\':
Packit 08bd4c
				c = '\\';
Packit 08bd4c
				++src;
Packit 08bd4c
				break;
Packit 08bd4c
			}
Packit 08bd4c
		}
Packit 08bd4c
		*dest++ = c;
Packit 08bd4c
	}
Packit 08bd4c
	*dest = '\0';
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/* Parse a hex digit. */
Packit 08bd4c
static int
Packit 08bd4c
parsedigit(char c)
Packit 08bd4c
{
Packit 08bd4c
	if (c >= '0' && c <= '9')
Packit 08bd4c
		return c - '0';
Packit 08bd4c
	else if (c >= 'a' && c <= 'f')
Packit 08bd4c
		return c - 'a';
Packit 08bd4c
	else if (c >= 'A' && c <= 'F')
Packit 08bd4c
		return c - 'A';
Packit 08bd4c
	else
Packit 08bd4c
		return -1;
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Note that this implementation does not (and should not!) obey
Packit 08bd4c
 * locale settings; you cannot simply substitute strtol here, since
Packit 08bd4c
 * it does obey locale.
Packit 08bd4c
 */
Packit 08bd4c
static int64_t
Packit 08bd4c
mtree_atol(char **p, int base)
Packit 08bd4c
{
Packit 08bd4c
	int64_t l, limit;
Packit 08bd4c
	int digit, last_digit_limit;
Packit 08bd4c
Packit 08bd4c
	if (base == 0) {
Packit 08bd4c
		if (**p != '0')
Packit 08bd4c
			base = 10;
Packit 08bd4c
		else if ((*p)[1] == 'x' || (*p)[1] == 'X') {
Packit 08bd4c
			*p += 2;
Packit 08bd4c
			base = 16;
Packit 08bd4c
		} else {
Packit 08bd4c
			base = 8;
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
Packit 08bd4c
	if (**p == '-') {
Packit 08bd4c
		limit = INT64_MIN / base;
Packit 08bd4c
		last_digit_limit = INT64_MIN % base;
Packit 08bd4c
		++(*p);
Packit 08bd4c
Packit 08bd4c
		l = 0;
Packit 08bd4c
		digit = parsedigit(**p);
Packit 08bd4c
		while (digit >= 0 && digit < base) {
Packit 08bd4c
			if (l < limit || (l == limit && digit > last_digit_limit))
Packit 08bd4c
				return INT64_MIN;
Packit 08bd4c
			l = (l * base) - digit;
Packit 08bd4c
			digit = parsedigit(*++(*p));
Packit 08bd4c
		}
Packit 08bd4c
		return l;
Packit 08bd4c
	} else {
Packit 08bd4c
		limit = INT64_MAX / base;
Packit 08bd4c
		last_digit_limit = INT64_MAX % base;
Packit 08bd4c
Packit 08bd4c
		l = 0;
Packit 08bd4c
		digit = parsedigit(**p);
Packit 08bd4c
		while (digit >= 0 && digit < base) {
Packit 08bd4c
			if (l > limit || (l == limit && digit > last_digit_limit))
Packit 08bd4c
				return INT64_MAX;
Packit 08bd4c
			l = (l * base) + digit;
Packit 08bd4c
			digit = parsedigit(*++(*p));
Packit 08bd4c
		}
Packit 08bd4c
		return l;
Packit 08bd4c
	}
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
/*
Packit 08bd4c
 * Returns length of line (including trailing newline)
Packit 08bd4c
 * or negative on error.  'start' argument is updated to
Packit 08bd4c
 * point to first character of line.
Packit 08bd4c
 */
Packit 08bd4c
static ssize_t
Packit 08bd4c
readline(struct archive_read *a, struct mtree *mtree, char **start,
Packit 08bd4c
    ssize_t limit)
Packit 08bd4c
{
Packit 08bd4c
	ssize_t bytes_read;
Packit 08bd4c
	ssize_t total_size = 0;
Packit 08bd4c
	ssize_t find_off = 0;
Packit 08bd4c
	const void *t;
Packit 08bd4c
	void *nl;
Packit 08bd4c
	char *u;
Packit 08bd4c
Packit 08bd4c
	/* Accumulate line in a line buffer. */
Packit 08bd4c
	for (;;) {
Packit 08bd4c
		/* Read some more. */
Packit 08bd4c
		t = __archive_read_ahead(a, 1, &bytes_read);
Packit 08bd4c
		if (t == NULL)
Packit 08bd4c
			return (0);
Packit 08bd4c
		if (bytes_read < 0)
Packit 08bd4c
			return (ARCHIVE_FATAL);
Packit 08bd4c
		nl = memchr(t, '\n', bytes_read);
Packit 08bd4c
		/* If we found '\n', trim the read to end exactly there. */
Packit 08bd4c
		if (nl != NULL) {
Packit 08bd4c
			bytes_read = ((const char *)nl) - ((const char *)t) + 1;
Packit 08bd4c
		}
Packit 08bd4c
		if (total_size + bytes_read + 1 > limit) {
Packit 08bd4c
			archive_set_error(&a->archive,
Packit 08bd4c
			    ARCHIVE_ERRNO_FILE_FORMAT,
Packit 08bd4c
			    "Line too long");
Packit 08bd4c
			return (ARCHIVE_FATAL);
Packit 08bd4c
		}
Packit 08bd4c
		if (archive_string_ensure(&mtree->line,
Packit 08bd4c
			total_size + bytes_read + 1) == NULL) {
Packit 08bd4c
			archive_set_error(&a->archive, ENOMEM,
Packit 08bd4c
			    "Can't allocate working buffer");
Packit 08bd4c
			return (ARCHIVE_FATAL);
Packit 08bd4c
		}
Packit 08bd4c
		/* Append new bytes to string. */
Packit 08bd4c
		memcpy(mtree->line.s + total_size, t, bytes_read);
Packit 08bd4c
		__archive_read_consume(a, bytes_read);
Packit 08bd4c
		total_size += bytes_read;
Packit 08bd4c
		mtree->line.s[total_size] = '\0';
Packit 08bd4c
Packit 08bd4c
		for (u = mtree->line.s + find_off; *u; ++u) {
Packit 08bd4c
			if (u[0] == '\n') {
Packit 08bd4c
				/* Ends with unescaped newline. */
Packit 08bd4c
				*start = mtree->line.s;
Packit 08bd4c
				return total_size;
Packit 08bd4c
			} else if (u[0] == '#') {
Packit 08bd4c
				/* Ends with comment sequence #...\n */
Packit 08bd4c
				if (nl == NULL) {
Packit 08bd4c
					/* But we've not found the \n yet */
Packit 08bd4c
					break;
Packit 08bd4c
				}
Packit 08bd4c
			} else if (u[0] == '\\') {
Packit 08bd4c
				if (u[1] == '\n') {
Packit 08bd4c
					/* Trim escaped newline. */
Packit 08bd4c
					total_size -= 2;
Packit 08bd4c
					mtree->line.s[total_size] = '\0';
Packit 08bd4c
					break;
Packit 08bd4c
				} else if (u[1] != '\0') {
Packit 08bd4c
					/* Skip the two-char escape sequence */
Packit 08bd4c
					++u;
Packit 08bd4c
				}
Packit 08bd4c
			}
Packit 08bd4c
		}
Packit 08bd4c
		find_off = u - mtree->line.s;
Packit 08bd4c
	}
Packit 08bd4c
}
Packit 08bd4c
Packit 08bd4c
static unsigned int
Packit 08bd4c
hash(const char *p)
Packit 08bd4c
{
Packit 08bd4c
	/* A 32-bit version of Peter Weinberger's (PJW) hash algorithm,
Packit 08bd4c
	   as used by ELF for hashing function names. */
Packit 08bd4c
	unsigned g, h = 0;
Packit 08bd4c
	while (*p != '\0') {
Packit 08bd4c
		h = (h << 4) + *p++;
Packit 08bd4c
		if ((g = h & 0xF0000000) != 0) {
Packit 08bd4c
			h ^= g >> 24;
Packit 08bd4c
			h &= 0x0FFFFFFF;
Packit 08bd4c
		}
Packit 08bd4c
	}
Packit 08bd4c
	return h;
Packit 08bd4c
}