1 /*
2 * Copyright (c) Wipro Technologies Ltd, 2002. All Rights Reserved.
3 *
4 * This program is free software; you can redistribute it and/or modify it
5 * under the terms of version 2 of the GNU General Public License as
6 * published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope that it would be useful, but
9 * WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
11 *
12 * You should have received a copy of the GNU General Public License along
13 * with this program; if not, write the Free Software Foundation, Inc.,
14 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
15 *
16 */
17 /**********************************************************
18 *
19 * TEST IDENTIFIER : ptrace03
20 *
21 * EXECUTED BY : anyone
22 *
23 * TEST TITLE : Tests for error conditions
24 *
25 * TEST CASE TOTAL : 3
26 *
27 * AUTHOR : Saji Kumar.V.R <saji.kumar@wipro.com>
28 *
29 * SIGNALS
30 * Uses SIGUSR1 to pause before test if option set.
31 * (See the parse_opts(3) man page).
32 *
33 * DESCRIPTION
34 * Verifies that
35 * 1) ptrace() returns -1 & sets errno to EPERM while tring to trace
36 * process 1
37 * (This test case will be executed only if the kernel version
38 * is 2.6.25 or below)
39 * 2) ptrace() returns -1 & sets errno to ESRCH if process with
40 * specified pid does not exist
41 * 3) ptrace() returns -1 & sets errno to EPERM if we are trying
42 * to trace a process which is already been traced
43 *
44 * Setup:
45 * Setup signal handling.
46 * Pause for SIGUSR1 if option specified.
47 *
48 * Test:
49 * Loop if the proper options are given.
50 * setup signal handler for SIGUSR2 signal
51 * fork a child
52 *
53 * CHILD:
54 * call ptrace() with proper arguments
55 * if ptrace() failed with expected return value & errno
56 * exit with errno
57 * else
58 * Give proper error message
59 * exit with errno
60 *
61 * PARENT:
62 * Wait for child to finish
63 * if child exits with expected errno
64 * Test Passed
65 * else
66 * Test failed
67 *
68 * Cleanup:
69 * Print errno log and/or timing stats if options given
70 *
71 * USAGE: <for command-line>
72 * ptrace03 [-c n] [-e] [-i n] [-I x] [-P x] [-t] [-h] [-f] [-p]
73 * where, -c n : Run n copies concurrently.
74 * -e : Turn on errno logging.
75 * -h : Show help screen
76 * -f : Turn off functional testing
77 * -i n : Execute test n times.
78 * -I x : Execute test for x seconds.
79 * -p : Pause for SIGUSR1 before starting
80 * -P x : Pause for x seconds between iterations.
81 * -t : Turn on syscall timing.
82 *
83 ****************************************************************/
84
85 #include <errno.h>
86 #include <signal.h>
87 #include <sys/wait.h>
88 #include <pwd.h>
89
90 #include <config.h>
91 #include "ptrace.h"
92
93 #include "test.h"
94
95 static void setup(void);
96 static void cleanup(void);
97
98 char *TCID = "ptrace03";
99
100 static pid_t init_pid = 1;
101 static pid_t unused_pid;
102 static pid_t zero_pid;
103
104 struct test_case_t {
105 int request;
106 pid_t *pid;
107 int exp_errno;
108 } test_cases[] = {
109 {
110 PTRACE_ATTACH, &init_pid, EPERM}, {
111 PTRACE_ATTACH, &unused_pid, ESRCH}, {
112 PTRACE_TRACEME, &zero_pid, EPERM},};
113
114 int TST_TOTAL = sizeof(test_cases) / sizeof(test_cases[0]);
115
main(int ac,char ** av)116 int main(int ac, char **av)
117 {
118
119 int lc, i;
120 pid_t child_pid;
121 int status;
122
123 tst_parse_opts(ac, av, NULL, NULL);
124
125 setup();
126
127 for (lc = 0; TEST_LOOPING(lc); lc++) {
128
129 tst_count = 0;
130
131 for (i = 0; i < TST_TOTAL; ++i) {
132
133 /* since Linux 2.6.26, it's allowed to trace init,
134 so just skip this test case */
135 if (i == 0 && tst_kvercmp(2, 6, 25) > 0) {
136 tst_resm(TCONF,
137 "this kernel allows to trace init");
138 continue;
139 }
140
141 /* fork() */
142 switch (child_pid = FORK_OR_VFORK()) {
143
144 case -1:
145 /* fork() failed */
146 tst_resm(TFAIL, "fork() failed");
147 continue;
148
149 case 0:
150 /* Child */
151
152 /* setup for third test case */
153 if (i == 2) {
154 if ((ptrace(PTRACE_TRACEME, 0,
155 NULL, NULL)) == -1) {
156 tst_resm(TWARN, "ptrace()"
157 " falied with errno, %d : %s",
158 errno,
159 strerror(errno));
160 exit(0);
161 }
162 }
163
164 TEST(ptrace(test_cases[i].request,
165 *(test_cases[i].pid), NULL, NULL));
166 if ((TEST_RETURN == -1) && (TEST_ERRNO ==
167 test_cases
168 [i].exp_errno)) {
169 exit(TEST_ERRNO);
170 } else {
171 tst_resm(TWARN | TTERRNO,
172 "ptrace() returned %ld",
173 TEST_RETURN);
174 exit(TEST_ERRNO);
175 }
176
177 default:
178 /* Parent */
179 if ((waitpid(child_pid, &status, 0)) < 0) {
180 tst_resm(TFAIL, "waitpid() failed");
181 continue;
182 }
183 if ((WIFEXITED(status)) &&
184 (WEXITSTATUS(status) ==
185 test_cases[i].exp_errno)) {
186 tst_resm(TPASS, "Test Passed");
187 } else {
188 tst_resm(TFAIL, "Test Failed");
189 }
190 }
191 }
192 }
193
194 /* cleanup and exit */
195 cleanup();
196
197 tst_exit();
198
199 }
200
201 /* setup() - performs all ONE TIME setup for this test */
setup(void)202 void setup(void)
203 {
204 unused_pid = tst_get_unused_pid(cleanup);
205
206 TEST_PAUSE;
207
208 }
209
210 /*
211 *cleanup() - performs all ONE TIME cleanup for this test at
212 * completion or premature exit.
213 */
cleanup(void)214 void cleanup(void)
215 {
216
217 }
218