source: trunk/essentials/app-shells/bash/builtins/jobs.def

Last change on this file was 3228, checked in by bird, 18 years ago

bash 3.1

File size: 6.9 KB
Line 
1This file is jobs.def, from which is created jobs.c.
2It implements the builtins "jobs" and "disown" in Bash.
3
4Copyright (C) 1987-2005 Free Software Foundation, Inc.
5
6This file is part of GNU Bash, the Bourne Again SHell.
7
8Bash is free software; you can redistribute it and/or modify it under
9the terms of the GNU General Public License as published by the Free
10Software Foundation; either version 2, or (at your option) any later
11version.
12
13Bash is distributed in the hope that it will be useful, but WITHOUT ANY
14WARRANTY; without even the implied warranty of MERCHANTABILITY or
15FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
16for more details.
17
18You should have received a copy of the GNU General Public License along
19with Bash; see the file COPYING. If not, write to the Free Software
20Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA.
21
22$PRODUCES jobs.c
23
24$BUILTIN jobs
25$FUNCTION jobs_builtin
26$DEPENDS_ON JOB_CONTROL
27$SHORT_DOC jobs [-lnprs] [jobspec ...] or jobs -x command [args]
28Lists the active jobs. The -l option lists process id's in addition
29to the normal information; the -p option lists process id's only.
30If -n is given, only processes that have changed status since the last
31notification are printed. JOBSPEC restricts output to that job. The
32-r and -s options restrict output to running and stopped jobs only,
33respectively. Without options, the status of all active jobs is
34printed. If -x is given, COMMAND is run after all job specifications
35that appear in ARGS have been replaced with the process ID of that job's
36process group leader.
37$END
38
39#include <config.h>
40
41#if defined (JOB_CONTROL)
42#include "../bashtypes.h"
43#include <signal.h>
44#if defined (HAVE_UNISTD_H)
45# include <unistd.h>
46#endif
47
48#include "../bashansi.h"
49#include "../bashintl.h"
50
51#include "../shell.h"
52#include "../jobs.h"
53#include "../execute_cmd.h"
54#include "bashgetopt.h"
55#include "common.h"
56
57#define JSTATE_ANY 0x0
58#define JSTATE_RUNNING 0x1
59#define JSTATE_STOPPED 0x2
60
61static int execute_list_with_replacements __P((WORD_LIST *));
62
63/* The `jobs' command. Prints outs a list of active jobs. If the
64 argument `-l' is given, then the process id's are printed also.
65 If the argument `-p' is given, print the process group leader's
66 pid only. If `-n' is given, only processes that have changed
67 status since the last notification are printed. If -x is given,
68 replace all job specs with the pid of the appropriate process
69 group leader and execute the command. The -r and -s options mean
70 to print info about running and stopped jobs only, respectively. */
71int
72jobs_builtin (list)
73 WORD_LIST *list;
74{
75 int form, execute, state, opt, any_failed, job;
76 sigset_t set, oset;
77
78 execute = any_failed = 0;
79 form = JLIST_STANDARD;
80 state = JSTATE_ANY;
81
82 reset_internal_getopt ();
83 while ((opt = internal_getopt (list, "lpnxrs")) != -1)
84 {
85 switch (opt)
86 {
87 case 'l':
88 form = JLIST_LONG;
89 break;
90 case 'p':
91 form = JLIST_PID_ONLY;
92 break;
93 case 'n':
94 form = JLIST_CHANGED_ONLY;
95 break;
96 case 'x':
97 if (form != JLIST_STANDARD)
98 {
99 builtin_error (_("no other options allowed with `-x'"));
100 return (EXECUTION_FAILURE);
101 }
102 execute++;
103 break;
104 case 'r':
105 state = JSTATE_RUNNING;
106 break;
107 case 's':
108 state = JSTATE_STOPPED;
109 break;
110
111 default:
112 builtin_usage ();
113 return (EX_USAGE);
114 }
115 }
116
117 list = loptend;
118
119 if (execute)
120 return (execute_list_with_replacements (list));
121
122 if (!list)
123 {
124 switch (state)
125 {
126 case JSTATE_ANY:
127 list_all_jobs (form);
128 break;
129 case JSTATE_RUNNING:
130 list_running_jobs (form);
131 break;
132 case JSTATE_STOPPED:
133 list_stopped_jobs (form);
134 break;
135 }
136 return (EXECUTION_SUCCESS);
137 }
138
139 while (list)
140 {
141 BLOCK_CHILD (set, oset);
142 job = get_job_spec (list);
143
144 if ((job == NO_JOB) || jobs == 0 || get_job_by_jid (job) == 0)
145 {
146 sh_badjob (list->word->word);
147 any_failed++;
148 }
149 else if (job != DUP_JOB)
150 list_one_job ((JOB *)NULL, form, 0, job);
151
152 UNBLOCK_CHILD (oset);
153 list = list->next;
154 }
155 return (any_failed ? EXECUTION_FAILURE : EXECUTION_SUCCESS);
156}
157
158static int
159execute_list_with_replacements (list)
160 WORD_LIST *list;
161{
162 register WORD_LIST *l;
163 int job, result;
164 COMMAND *command;
165 JOB *j;
166
167 /* First do the replacement of job specifications with pids. */
168 for (l = list; l; l = l->next)
169 {
170 if (l->word->word[0] == '%') /* we have a winner */
171 {
172 job = get_job_spec (l);
173
174 /* A bad job spec is not really a job spec! Pass it through. */
175 if (INVALID_JOB (job))
176 continue;
177
178 j = get_job_by_jid (job);
179 free (l->word->word);
180 l->word->word = itos (j->pgrp);
181 }
182 }
183
184 /* Next make a new simple command and execute it. */
185 begin_unwind_frame ("jobs_builtin");
186
187 command = make_bare_simple_command ();
188 command->value.Simple->words = copy_word_list (list);
189 command->value.Simple->redirects = (REDIRECT *)NULL;
190 command->flags |= CMD_INHIBIT_EXPANSION;
191 command->value.Simple->flags |= CMD_INHIBIT_EXPANSION;
192
193 add_unwind_protect (dispose_command, command);
194 result = execute_command (command);
195 dispose_command (command);
196
197 discard_unwind_frame ("jobs_builtin");
198 return (result);
199}
200#endif /* JOB_CONTROL */
201
202$BUILTIN disown
203$FUNCTION disown_builtin
204$DEPENDS_ON JOB_CONTROL
205$SHORT_DOC disown [-h] [-ar] [jobspec ...]
206By default, removes each JOBSPEC argument from the table of active jobs.
207If the -h option is given, the job is not removed from the table, but is
208marked so that SIGHUP is not sent to the job if the shell receives a
209SIGHUP. The -a option, when JOBSPEC is not supplied, means to remove all
210jobs from the job table; the -r option means to remove only running jobs.
211$END
212
213#if defined (JOB_CONTROL)
214int
215disown_builtin (list)
216 WORD_LIST *list;
217{
218 int opt, job, retval, nohup_only, running_jobs, all_jobs;
219 sigset_t set, oset;
220 intmax_t pid_value;
221
222 nohup_only = running_jobs = all_jobs = 0;
223 reset_internal_getopt ();
224 while ((opt = internal_getopt (list, "ahr")) != -1)
225 {
226 switch (opt)
227 {
228 case 'a':
229 all_jobs = 1;
230 break;
231 case 'h':
232 nohup_only = 1;
233 break;
234 case 'r':
235 running_jobs = 1;
236 break;
237 default:
238 builtin_usage ();
239 return (EX_USAGE);
240 }
241 }
242 list = loptend;
243 retval = EXECUTION_SUCCESS;
244
245 /* `disown -a' or `disown -r' */
246 if (list == 0 && (all_jobs || running_jobs))
247 {
248 if (nohup_only)
249 nohup_all_jobs (running_jobs);
250 else
251 delete_all_jobs (running_jobs);
252 return (EXECUTION_SUCCESS);
253 }
254
255 do
256 {
257 BLOCK_CHILD (set, oset);
258 job = (list && legal_number (list->word->word, &pid_value) && pid_value == (pid_t) pid_value)
259 ? get_job_by_pid ((pid_t) pid_value, 0)
260 : get_job_spec (list);
261
262 if (job == NO_JOB || jobs == 0 || INVALID_JOB (job))
263 {
264 sh_badjob (list ? list->word->word : "current");
265 retval = EXECUTION_FAILURE;
266 }
267 else if (nohup_only)
268 nohup_job (job);
269 else
270 delete_job (job, 1);
271 UNBLOCK_CHILD (oset);
272
273 if (list)
274 list = list->next;
275 }
276 while (list);
277
278 return (retval);
279}
280#endif /* JOB_CONTROL */
Note: See TracBrowser for help on using the repository browser.