Files
tar/tests/checkseekhole.c
Paul Eggert 281e03ec6c Prefer < 0 to == -1 where either will do
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.
2024-08-04 01:41:43 -07:00

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
}