New file
This commit is contained in:
295
src/xheader.c
Normal file
295
src/xheader.c
Normal file
@@ -0,0 +1,295 @@
|
||||
/* This file is part of GNU Tar
|
||||
|
||||
Copyright (C) 2003 Free Software Foundation, Inc.
|
||||
|
||||
This program is free software; you can redistribute it and/or modify it
|
||||
under the terms of the GNU General Public License as published by the
|
||||
Free Software Foundation; either version 2, or (at your option) any later
|
||||
version.
|
||||
|
||||
This program is distributed in the hope that it will be useful, but
|
||||
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
|
||||
Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License along
|
||||
with this program; if not, write to the Free Software Foundation, Inc.,
|
||||
59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
|
||||
|
||||
#include "system.h"
|
||||
|
||||
#include <grp.h>
|
||||
#include <hash.h>
|
||||
#include <pwd.h>
|
||||
#include <quotearg.h>
|
||||
|
||||
#include "common.h"
|
||||
|
||||
/* Forward declarations */
|
||||
static void dummy_handler (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void atime_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void atime_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void gid_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void gid_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void gname_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void gname_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void linkpath_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void linkpath_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void mtime_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void mtime_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void ctime_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void ctime_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void path_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void path_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void size_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void size_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void uid_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void uid_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
static void uname_coder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
static void uname_decoder (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
|
||||
/* General Interface */
|
||||
|
||||
struct xhdr_tab
|
||||
{
|
||||
char *keyword;
|
||||
void (*coder) (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
void (*decoder) (struct tar_stat_info *st, char *keyword, char *arg);
|
||||
};
|
||||
|
||||
struct xhdr_tab xhdr_tab[] = {
|
||||
{ "atime", atime_coder, atime_decoder },
|
||||
{ "comment", dummy_handler, dummy_handler },
|
||||
{ "charset", dummy_handler, dummy_handler },
|
||||
{ "ctime", ctime_coder, ctime_decoder },
|
||||
{ "gid", gid_coder, gid_decoder },
|
||||
{ "gname", gname_coder, gname_decoder },
|
||||
{ "linkpath", linkpath_coder, linkpath_decoder },
|
||||
{ "mtime", mtime_coder, mtime_decoder },
|
||||
{ "path", path_coder, path_decoder },
|
||||
{ "size", size_coder, size_decoder },
|
||||
{ "uid", uid_coder, uid_decoder },
|
||||
{ "uname", uname_coder, uname_decoder },
|
||||
{ NULL },
|
||||
};
|
||||
|
||||
static struct xhdr_tab *
|
||||
locate_handler (char *keyword)
|
||||
{
|
||||
struct xhdr_tab *p;
|
||||
|
||||
for (p = xhdr_tab; p->keyword; p++)
|
||||
if (strcmp (p->keyword, keyword) == 0)
|
||||
return p;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static int
|
||||
decode_record (char **p, struct tar_stat_info *st)
|
||||
{
|
||||
size_t len;
|
||||
char *keyword, *eqp;
|
||||
char *start = *p;
|
||||
struct xhdr_tab *t;
|
||||
|
||||
if (**p == 0)
|
||||
return 1;
|
||||
|
||||
len = strtoul (*p, p, 10);
|
||||
if (**p != ' ')
|
||||
{
|
||||
ERROR ((0, 0, _("Malformed extended headed")));
|
||||
return 1;
|
||||
}
|
||||
|
||||
keyword = ++*p;
|
||||
for (;*p < start + len; ++*p)
|
||||
if (**p == '=')
|
||||
break;
|
||||
|
||||
if (**p != '=')
|
||||
{
|
||||
ERROR ((0, 0, _("Malformed extended headed")));
|
||||
return 1;
|
||||
}
|
||||
|
||||
eqp = *p;
|
||||
**p = 0;
|
||||
t = locate_handler (keyword);
|
||||
if (t)
|
||||
{
|
||||
char endc;
|
||||
char *value;
|
||||
|
||||
value = ++*p;
|
||||
|
||||
endc = start[len-1];
|
||||
start[len-1] = 0;
|
||||
t->decoder (st, keyword, value);
|
||||
start[len-1] = endc;
|
||||
}
|
||||
*eqp = '=';
|
||||
*p = &start[len];
|
||||
return 0;
|
||||
}
|
||||
|
||||
void
|
||||
xheader_decode (struct tar_stat_info *st)
|
||||
{
|
||||
char *p, *endp;
|
||||
|
||||
p = extended_header.blocks->buffer;
|
||||
endp = &extended_header.blocks[extended_header.nblocks-1].buffer
|
||||
[sizeof(extended_header.blocks[0])-1];
|
||||
while (p < endp)
|
||||
if (decode_record (&p, st))
|
||||
break;
|
||||
}
|
||||
|
||||
void
|
||||
xheader_read (union block *p, size_t size)
|
||||
{
|
||||
size_t i;
|
||||
|
||||
free (extended_header.blocks);
|
||||
extended_header.nblocks = (size + BLOCKSIZE - 1) / BLOCKSIZE;
|
||||
extended_header.blocks = xmalloc (sizeof (extended_header.blocks[0]) *
|
||||
extended_header.nblocks);
|
||||
set_next_block_after (p);
|
||||
for (i = 0; i < extended_header.nblocks; i++)
|
||||
{
|
||||
p = find_next_block ();
|
||||
memcpy (&extended_header.blocks[i], p, sizeof (p[0]));
|
||||
set_next_block_after (p);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/* Implementations */
|
||||
static void
|
||||
dummy_handler (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
atime_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
atime_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->stat.st_atime = strtoul (arg, NULL, 0);
|
||||
}
|
||||
|
||||
static void
|
||||
gid_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
gid_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->stat.st_gid = strtoul (arg, NULL, 0);
|
||||
}
|
||||
|
||||
static void
|
||||
gname_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
gname_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->gname = strdup (arg);
|
||||
}
|
||||
|
||||
static void
|
||||
linkpath_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
linkpath_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->link_name = strdup (arg);
|
||||
}
|
||||
|
||||
static void
|
||||
ctime_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
ctime_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->stat.st_ctime = strtoul (arg, NULL, 0);
|
||||
}
|
||||
|
||||
static void
|
||||
mtime_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
mtime_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->stat.st_mtime = strtoul (arg, NULL, 0);
|
||||
}
|
||||
|
||||
static void
|
||||
path_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
path_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
assign_string (&st->orig_file_name, arg);
|
||||
assign_string (&st->file_name, arg);
|
||||
st->had_trailing_slash = strip_trailing_slashes (st->file_name);
|
||||
}
|
||||
|
||||
static void
|
||||
size_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
size_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->stat.st_size = strtoul (arg, NULL, 0);
|
||||
}
|
||||
|
||||
static void
|
||||
uid_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
uid_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->stat.st_uid = strtoul (arg, NULL, 0);
|
||||
}
|
||||
|
||||
static void
|
||||
uname_coder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
uname_decoder (struct tar_stat_info *st, char *keyword, char *arg)
|
||||
{
|
||||
st->uname = strdup (arg);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user