1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
|
#!/bin/sh
# Test df's behaviour when the mount list cannot be read.
# This test is skipped on systems that lack LD_PRELOAD support; that's fine.
# Copyright (C) 2012-2022 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 of the License, 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 <https://www.gnu.org/licenses/>.
. "${srcdir=.}/tests/init.sh"; path_prepend_ ./src
print_ver_ df
require_gcc_shared_
# Protect against inaccessible remote mounts etc.
timeout 10 df || skip_ "df fails"
grep '^#define HAVE_MNTENT_H 1' $CONFIG_HEADER > /dev/null \
|| skip_ "no mntent.h available to confirm the interface"
grep '^#define HAVE_GETMNTENT 1' $CONFIG_HEADER > /dev/null \
|| skip_ "getmntent is not used on this system"
# Simulate "mtab" failure.
cat > k.c <<EOF || framework_failure_
#define _GNU_SOURCE
#include <stdio.h>
#include <errno.h>
#include <mntent.h>
#include <string.h>
#include <dlfcn.h>
#define STREQ(a, b) (strcmp (a, b) == 0)
FILE* fopen(const char *path, const char *mode)
{
static FILE* (*fopen_func)(char const *, char const *);
/* get reference to original (libc provided) fopen */
if (!fopen_func)
{
fopen_func = (FILE*(*)(char const *, char const *))
dlsym(RTLD_NEXT, "fopen");
if (!fopen_func)
{
fprintf (stderr, "Failed to find fopen()\n");
errno = ESRCH;
return NULL;
}
}
/* Returning ENOENT here will get read_file_system_list()
to fall back to using getmntent() below. */
if (STREQ (path, "/proc/self/mountinfo"))
{
errno = ENOENT;
return NULL;
}
else
return fopen_func(path, mode);
}
struct mntent *getmntent (FILE *fp)
{
/* Prove that LD_PRELOAD works. */
static int done = 0;
if (!done)
{
fclose (fopen ("x", "w"));
++done;
}
/* Now simulate the failure. */
errno = ENOENT;
return NULL;
}
EOF
# Then compile/link it:
gcc_shared_ k.c k.so \
|| framework_failure_ 'failed to build shared library'
cleanup_() { unset LD_PRELOAD; }
export LD_PRELOAD=$LD_PRELOAD:./k.so
# Test if LD_PRELOAD works:
df 2>/dev/null
test -f x || skip_ "internal test failure: maybe LD_PRELOAD doesn't work?"
# These tests are supposed to succeed:
df '.' || fail=1
df -i '.' || fail=1
df -T '.' || fail=1
df -Ti '.' || fail=1
df --total '.' || fail=1
# These tests are supposed to fail:
returns_ 1 df || fail=1
returns_ 1 df -i || fail=1
returns_ 1 df -T || fail=1
returns_ 1 df -Ti || fail=1
returns_ 1 df --total || fail=1
returns_ 1 df -a || fail=1
returns_ 1 df -a '.' || fail=1
returns_ 1 df -l || fail=1
returns_ 1 df -l '.' || fail=1
returns_ 1 df -t hello || fail=1
returns_ 1 df -t hello '.' || fail=1
returns_ 1 df -x hello || fail=1
returns_ 1 df -x hello '.' || fail=1
Exit $fail
|