Also, fix an unlikely read overflow in sys_exec_setmtime_script. * src/buffer.c (open_compressed_archive): * src/compare.c (verify_volume): * src/exclist.c (info_attach_exclist): * src/misc.c (xfork): * src/sparse.c (sparse_scan_file_seek): * src/system.c (sys_wait_for_child, sys_spawn_shell) (wait_for_grandchild, sys_wait_command, sys_exec_info_script) (sys_exec_checkpoint_script, sys_exec_setmtime_script): * src/transform.c (_single_transform_name_to_obstack): * src/xattrs.c (xattrs__acls_set, xattrs_acls_get) (xattrs_xattrs_get, xattrs__fd_set, xattrs_selinux_get) (xattrs_selinux_set): * tests/checkseekhole.c (check_seek_hole, main): Simplify failure tests by just looking at return value sign. * src/system.c (sys_exec_setmtime_script): Don’t assume ‘read’ result fits in int. (sys_exec_setmtime_script): Don’t reject 1 second before Epoch.
93 lines
2.0 KiB
C
93 lines
2.0 KiB
C
/* Test suite for GNU tar - SEEK_HOLE detector.
|
|
|
|
Copyright 2015-2024 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 3, 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
Description: detect whether it is possible to work with SEEK_HOLE on
|
|
particular operating system and file system. */
|
|
|
|
#include "config.h"
|
|
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <fcntl.h>
|
|
|
|
enum {
|
|
EX_OK = 0, /* SEEK_HOLE support */
|
|
EX_FAIL, /* test failed - no SEEK_HOLE support */
|
|
EX_BAD, /* test is not relevant */
|
|
};
|
|
|
|
int
|
|
check_seek_hole (int fd)
|
|
{
|
|
#ifdef SEEK_HOLE
|
|
struct stat stat;
|
|
off_t offset;
|
|
|
|
/* hole of 100MB */
|
|
if (lseek (fd, 100*1024*1024, SEEK_END) < 0)
|
|
return EX_BAD;
|
|
|
|
/* piece of data */
|
|
if (write (fd, "data\n", 5) != 5)
|
|
return EX_BAD;
|
|
|
|
/* another hole */
|
|
if (lseek (fd, 100*1024*1024, SEEK_END) < 0)
|
|
return EX_BAD;
|
|
|
|
/* piece of data */
|
|
if (write (fd, "data\n", 5) != 5)
|
|
return EX_BAD;
|
|
|
|
if (fstat (fd, &stat))
|
|
return EX_BAD;
|
|
|
|
offset = lseek (fd, 0, SEEK_DATA);
|
|
if (offset < 0)
|
|
return EX_FAIL;
|
|
|
|
offset = lseek (fd, offset, SEEK_HOLE);
|
|
if (offset < 0 || offset == stat.st_size)
|
|
return EX_FAIL;
|
|
|
|
return EX_OK;
|
|
#else
|
|
return EX_BAD;
|
|
#endif
|
|
}
|
|
|
|
int
|
|
main ()
|
|
{
|
|
#ifdef SEEK_HOLE
|
|
int rc;
|
|
char template[] = "testseekhole-XXXXXX";
|
|
int fd = mkstemp (template);
|
|
if (fd < 0)
|
|
return EX_BAD;
|
|
rc = check_seek_hole (fd);
|
|
close (fd);
|
|
unlink (template);
|
|
|
|
return rc;
|
|
#else
|
|
return EX_FAIL;
|
|
#endif
|
|
}
|