diff options
Diffstat (limited to 'arch/um/os-Linux/helper.c')
-rw-r--r-- | arch/um/os-Linux/helper.c | 38 |
1 files changed, 27 insertions, 11 deletions
diff --git a/arch/um/os-Linux/helper.c b/arch/um/os-Linux/helper.c index 7a72dbb61b0d..fba3f0fefeef 100644 --- a/arch/um/os-Linux/helper.c +++ b/arch/um/os-Linux/helper.c | |||
@@ -76,7 +76,7 @@ int run_helper(void (*pre_exec)(void *), void *pre_data, char **argv) | |||
76 | data.fd = fds[1]; | 76 | data.fd = fds[1]; |
77 | data.buf = __cant_sleep() ? kmalloc(PATH_MAX, UM_GFP_ATOMIC) : | 77 | data.buf = __cant_sleep() ? kmalloc(PATH_MAX, UM_GFP_ATOMIC) : |
78 | kmalloc(PATH_MAX, UM_GFP_KERNEL); | 78 | kmalloc(PATH_MAX, UM_GFP_KERNEL); |
79 | pid = clone(helper_child, (void *) sp, CLONE_VM | SIGCHLD, &data); | 79 | pid = clone(helper_child, (void *) sp, CLONE_VM, &data); |
80 | if (pid < 0) { | 80 | if (pid < 0) { |
81 | ret = -errno; | 81 | ret = -errno; |
82 | printk("run_helper : clone failed, errno = %d\n", errno); | 82 | printk("run_helper : clone failed, errno = %d\n", errno); |
@@ -101,7 +101,7 @@ int run_helper(void (*pre_exec)(void *), void *pre_data, char **argv) | |||
101 | ret = n; | 101 | ret = n; |
102 | kill(pid, SIGKILL); | 102 | kill(pid, SIGKILL); |
103 | } | 103 | } |
104 | CATCH_EINTR(waitpid(pid, NULL, 0)); | 104 | CATCH_EINTR(waitpid(pid, NULL, __WCLONE)); |
105 | } | 105 | } |
106 | 106 | ||
107 | out_free2: | 107 | out_free2: |
@@ -126,7 +126,7 @@ int run_helper_thread(int (*proc)(void *), void *arg, unsigned int flags, | |||
126 | return -ENOMEM; | 126 | return -ENOMEM; |
127 | 127 | ||
128 | sp = stack + UM_KERN_PAGE_SIZE - sizeof(void *); | 128 | sp = stack + UM_KERN_PAGE_SIZE - sizeof(void *); |
129 | pid = clone(proc, (void *) sp, flags | SIGCHLD, arg); | 129 | pid = clone(proc, (void *) sp, flags, arg); |
130 | if (pid < 0) { | 130 | if (pid < 0) { |
131 | err = -errno; | 131 | err = -errno; |
132 | printk("run_helper_thread : clone failed, errno = %d\n", | 132 | printk("run_helper_thread : clone failed, errno = %d\n", |
@@ -134,7 +134,7 @@ int run_helper_thread(int (*proc)(void *), void *arg, unsigned int flags, | |||
134 | return err; | 134 | return err; |
135 | } | 135 | } |
136 | if (stack_out == NULL) { | 136 | if (stack_out == NULL) { |
137 | CATCH_EINTR(pid = waitpid(pid, &status, 0)); | 137 | CATCH_EINTR(pid = waitpid(pid, &status, __WCLONE)); |
138 | if (pid < 0) { | 138 | if (pid < 0) { |
139 | err = -errno; | 139 | err = -errno; |
140 | printk("run_helper_thread - wait failed, errno = %d\n", | 140 | printk("run_helper_thread - wait failed, errno = %d\n", |
@@ -150,14 +150,30 @@ int run_helper_thread(int (*proc)(void *), void *arg, unsigned int flags, | |||
150 | return pid; | 150 | return pid; |
151 | } | 151 | } |
152 | 152 | ||
153 | int helper_wait(int pid) | 153 | int helper_wait(int pid, int nohang, char *pname) |
154 | { | 154 | { |
155 | int ret; | 155 | int ret, status; |
156 | int wflags = __WCLONE; | ||
156 | 157 | ||
157 | CATCH_EINTR(ret = waitpid(pid, NULL, WNOHANG)); | 158 | if (nohang) |
159 | wflags |= WNOHANG; | ||
160 | |||
161 | if (!pname) | ||
162 | pname = "helper_wait"; | ||
163 | |||
164 | CATCH_EINTR(ret = waitpid(pid, &status, wflags)); | ||
158 | if (ret < 0) { | 165 | if (ret < 0) { |
159 | ret = -errno; | 166 | printk(UM_KERN_ERR "%s : waitpid process %d failed, " |
160 | printk("helper_wait : waitpid failed, errno = %d\n", errno); | 167 | "errno = %d\n", pname, pid, errno); |
161 | } | 168 | return -errno; |
162 | return ret; | 169 | } else if (nohang && ret == 0) { |
170 | printk(UM_KERN_ERR "%s : process %d has not exited\n", | ||
171 | pname, pid); | ||
172 | return -ECHILD; | ||
173 | } else if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) { | ||
174 | printk(UM_KERN_ERR "%s : process %d didn't exit with " | ||
175 | "status 0\n", pname, pid); | ||
176 | return -ECHILD; | ||
177 | } else | ||
178 | return 0; | ||
163 | } | 179 | } |