OpenCores
URL https://opencores.org/ocsvn/or1k_soc_on_altera_embedded_dev_kit/or1k_soc_on_altera_embedded_dev_kit/trunk

Subversion Repositories or1k_soc_on_altera_embedded_dev_kit

[/] [or1k_soc_on_altera_embedded_dev_kit/] [trunk/] [linux-2.6/] [linux-2.6.24/] [arch/] [sparc64/] [solaris/] [signal.c] - Blame information for rev 3

Details | Compare with Previous | View Log

Line No. Rev Author Line
1 3 xianfeng
/* $Id: signal.c,v 1.7 2000/09/05 21:44:54 davem Exp $
2
 * signal.c: Signal emulation for Solaris
3
 *
4
 * Copyright (C) 1997 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
5
 */
6
 
7
#include <linux/types.h>
8
#include <linux/errno.h>
9
 
10
#include <asm/uaccess.h>
11
#include <asm/svr4.h>
12
#include <asm/string.h>
13
 
14
#include "conv.h"
15
#include "signal.h"
16
 
17
#define _S(nr) (1L<<((nr)-1))
18
 
19
#define _BLOCKABLE (~(_S(SIGKILL) | _S(SIGSTOP)))
20
 
21
long linux_to_solaris_signals[] = {
22
        0,
23
        SOLARIS_SIGHUP,         SOLARIS_SIGINT,
24
        SOLARIS_SIGQUIT,        SOLARIS_SIGILL,
25
        SOLARIS_SIGTRAP,        SOLARIS_SIGIOT,
26
        SOLARIS_SIGEMT,         SOLARIS_SIGFPE,
27
        SOLARIS_SIGKILL,        SOLARIS_SIGBUS,
28
        SOLARIS_SIGSEGV,        SOLARIS_SIGSYS,
29
        SOLARIS_SIGPIPE,        SOLARIS_SIGALRM,
30
        SOLARIS_SIGTERM,        SOLARIS_SIGURG,
31
        SOLARIS_SIGSTOP,        SOLARIS_SIGTSTP,
32
        SOLARIS_SIGCONT,        SOLARIS_SIGCLD,
33
        SOLARIS_SIGTTIN,        SOLARIS_SIGTTOU,
34
        SOLARIS_SIGPOLL,        SOLARIS_SIGXCPU,
35
        SOLARIS_SIGXFSZ,        SOLARIS_SIGVTALRM,
36
        SOLARIS_SIGPROF,        SOLARIS_SIGWINCH,
37
        SOLARIS_SIGUSR1,        SOLARIS_SIGUSR1,
38
        SOLARIS_SIGUSR2,        -1,
39
};
40
 
41
long solaris_to_linux_signals[] = {
42
        0,
43
        SIGHUP,         SIGINT,         SIGQUIT,        SIGILL,
44
        SIGTRAP,        SIGIOT,         SIGEMT,         SIGFPE,
45
        SIGKILL,        SIGBUS,         SIGSEGV,        SIGSYS,
46
        SIGPIPE,        SIGALRM,        SIGTERM,        SIGUSR1,
47
        SIGUSR2,        SIGCHLD,        -1,             SIGWINCH,
48
        SIGURG,         SIGPOLL,        SIGSTOP,        SIGTSTP,
49
        SIGCONT,        SIGTTIN,        SIGTTOU,        SIGVTALRM,
50
        SIGPROF,        SIGXCPU,        SIGXFSZ,        -1,
51
        -1,             -1,             -1,             -1,
52
        -1,             -1,             -1,             -1,
53
        -1,             -1,             -1,             -1,
54
};
55
 
56
static inline long mapsig(long sig)
57
{
58
        if ((unsigned long)sig > SOLARIS_NSIGNALS)
59
                return -EINVAL;
60
        return solaris_to_linux_signals[sig];
61
}
62
 
63
asmlinkage int solaris_kill(int pid, int sig)
64
{
65
        int (*sys_kill)(int,int) =
66
                (int (*)(int,int))SYS(kill);
67
        int s = mapsig(sig);
68
 
69
        if (s < 0) return s;
70
        return sys_kill(pid, s);
71
}
72
 
73
static long sig_handler(int sig, u32 arg, int one_shot)
74
{
75
        struct sigaction sa, old;
76
        int ret;
77
        mm_segment_t old_fs = get_fs();
78
        int (*sys_sigaction)(int,struct sigaction __user *,struct sigaction __user *) =
79
                (int (*)(int,struct sigaction __user *,struct sigaction __user *))SYS(sigaction);
80
 
81
        sigemptyset(&sa.sa_mask);
82
        sa.sa_restorer = NULL;
83
        sa.sa_handler = (__sighandler_t)A(arg);
84
        sa.sa_flags = 0;
85
        if (one_shot) sa.sa_flags = SA_ONESHOT | SA_NOMASK;
86
        set_fs (KERNEL_DS);
87
        ret = sys_sigaction(sig, (void __user *)&sa, (void __user *)&old);
88
        set_fs (old_fs);
89
        if (ret < 0) return ret;
90
        return (u32)(unsigned long)old.sa_handler;
91
}
92
 
93
static inline long solaris_signal(int sig, u32 arg)
94
{
95
        return sig_handler (sig, arg, 1);
96
}
97
 
98
static long solaris_sigset(int sig, u32 arg)
99
{
100
        if (arg != 2) /* HOLD */ {
101
                spin_lock_irq(&current->sighand->siglock);
102
                sigdelsetmask(&current->blocked, _S(sig));
103
                recalc_sigpending();
104
                spin_unlock_irq(&current->sighand->siglock);
105
                return sig_handler (sig, arg, 0);
106
        } else {
107
                spin_lock_irq(&current->sighand->siglock);
108
                sigaddsetmask(&current->blocked, (_S(sig) & ~_BLOCKABLE));
109
                recalc_sigpending();
110
                spin_unlock_irq(&current->sighand->siglock);
111
                return 0;
112
        }
113
}
114
 
115
static inline long solaris_sighold(int sig)
116
{
117
        return solaris_sigset(sig, 2);
118
}
119
 
120
static inline long solaris_sigrelse(int sig)
121
{
122
        spin_lock_irq(&current->sighand->siglock);
123
        sigdelsetmask(&current->blocked, _S(sig));
124
        recalc_sigpending();
125
        spin_unlock_irq(&current->sighand->siglock);
126
        return 0;
127
}
128
 
129
static inline long solaris_sigignore(int sig)
130
{
131
        return sig_handler(sig, (u32)(unsigned long)SIG_IGN, 0);
132
}
133
 
134
static inline long solaris_sigpause(int sig)
135
{
136
        printk ("Need to support solaris sigpause\n");
137
        return -ENOSYS;
138
}
139
 
140
asmlinkage long solaris_sigfunc(int sig, u32 arg)
141
{
142
        int func = sig & ~0xff;
143
 
144
        sig = mapsig(sig & 0xff);
145
        if (sig < 0) return sig;
146
        switch (func) {
147
        case 0: return solaris_signal(sig, arg);
148
        case 0x100: return solaris_sigset(sig, arg);
149
        case 0x200: return solaris_sighold(sig);
150
        case 0x400: return solaris_sigrelse(sig);
151
        case 0x800: return solaris_sigignore(sig);
152
        case 0x1000: return solaris_sigpause(sig);
153
        }
154
        return -EINVAL;
155
}
156
 
157
typedef struct {
158
        u32 __sigbits[4];
159
} sol_sigset_t;
160
 
161
static inline int mapin(u32 *p, sigset_t *q)
162
{
163
        int i;
164
        u32 x;
165
        int sig;
166
 
167
        sigemptyset(q);
168
        x = p[0];
169
        for (i = 1; i <= SOLARIS_NSIGNALS; i++) {
170
                if (x & 1) {
171
                        sig = solaris_to_linux_signals[i];
172
                        if (sig == -1)
173
                                return -EINVAL;
174
                        sigaddsetmask(q, (1L << (sig - 1)));
175
                }
176
                x >>= 1;
177
                if (i == 32)
178
                        x = p[1];
179
        }
180
        return 0;
181
}
182
 
183
static inline int mapout(sigset_t *q, u32 *p)
184
{
185
        int i;
186
        int sig;
187
 
188
        p[0] = 0;
189
        p[1] = 0;
190
        for (i = 1; i <= 32; i++) {
191
                if (sigismember(q, sigmask(i))) {
192
                        sig = linux_to_solaris_signals[i];
193
                        if (sig == -1)
194
                                return -EINVAL;
195
                        if (sig > 32)
196
                                p[1] |= 1L << (sig - 33);
197
                        else
198
                                p[0] |= 1L << (sig - 1);
199
                }
200
        }
201
        return 0;
202
}
203
 
204
asmlinkage int solaris_sigprocmask(int how, u32 in, u32 out)
205
{
206
        sigset_t in_s, *ins, out_s, *outs;
207
        mm_segment_t old_fs = get_fs();
208
        int ret;
209
        int (*sys_sigprocmask)(int,sigset_t __user *,sigset_t __user *) =
210
                (int (*)(int,sigset_t __user *,sigset_t __user *))SYS(sigprocmask);
211
 
212
        ins = NULL; outs = NULL;
213
        if (in) {
214
                u32 tmp[2];
215
 
216
                if (copy_from_user (tmp, (void __user *)A(in), 2*sizeof(u32)))
217
                        return -EFAULT;
218
                ins = &in_s;
219
                if (mapin (tmp, ins)) return -EINVAL;
220
        }
221
        if (out) outs = &out_s;
222
        set_fs (KERNEL_DS);
223
        ret = sys_sigprocmask((how == 3) ? SIG_SETMASK : how,
224
                                (void __user *)ins, (void __user *)outs);
225
        set_fs (old_fs);
226
        if (ret) return ret;
227
        if (out) {
228
                u32 tmp[4];
229
 
230
                tmp[2] = 0; tmp[3] = 0;
231
                if (mapout (outs, tmp)) return -EINVAL;
232
                if (copy_to_user((void __user *)A(out), tmp, 4*sizeof(u32)))
233
                        return -EFAULT;
234
        }
235
        return 0;
236
}
237
 
238
asmlinkage long do_sol_sigsuspend(u32 mask)
239
{
240
        sigset_t s;
241
        u32 tmp[2];
242
 
243
        if (copy_from_user (tmp, (sol_sigset_t __user *)A(mask), 2*sizeof(u32)))
244
                return -EFAULT;
245
        if (mapin (tmp, &s)) return -EINVAL;
246
        return (long)s.sig[0];
247
}
248
 
249
struct sol_sigaction {
250
        int     sa_flags;
251
        u32     sa_handler;
252
        u32     sa_mask[4];
253
        int     sa_resv[2];
254
};
255
 
256
asmlinkage int solaris_sigaction(int sig, u32 act, u32 old)
257
{
258
        u32 tmp, tmp2[4];
259
        struct sigaction s, s2;
260
        int ret;
261
        mm_segment_t old_fs = get_fs();
262
        struct sol_sigaction __user *p = (void __user *)A(old);
263
        int (*sys_sigaction)(int,struct sigaction __user *,struct sigaction __user *) =
264
                (int (*)(int,struct sigaction __user *,struct sigaction __user *))SYS(sigaction);
265
 
266
        sig = mapsig(sig);
267
        if (sig < 0) {
268
                /* We cheat a little bit for Solaris only signals */
269
                if (old && clear_user(p, sizeof(struct sol_sigaction)))
270
                        return -EFAULT;
271
                return 0;
272
        }
273
        if (act) {
274
                if (get_user (tmp, &p->sa_flags))
275
                        return -EFAULT;
276
                s.sa_flags = 0;
277
                if (tmp & SOLARIS_SA_ONSTACK) s.sa_flags |= SA_STACK;
278
                if (tmp & SOLARIS_SA_RESTART) s.sa_flags |= SA_RESTART;
279
                if (tmp & SOLARIS_SA_NODEFER) s.sa_flags |= SA_NOMASK;
280
                if (tmp & SOLARIS_SA_RESETHAND) s.sa_flags |= SA_ONESHOT;
281
                if (tmp & SOLARIS_SA_NOCLDSTOP) s.sa_flags |= SA_NOCLDSTOP;
282
                if (get_user (tmp, &p->sa_handler) ||
283
                    copy_from_user (tmp2, &p->sa_mask, 2*sizeof(u32)))
284
                        return -EFAULT;
285
                s.sa_handler = (__sighandler_t)A(tmp);
286
                if (mapin (tmp2, &s.sa_mask)) return -EINVAL;
287
                s.sa_restorer = NULL;
288
        }
289
        set_fs(KERNEL_DS);
290
        ret = sys_sigaction(sig, act ? (void __user *)&s : NULL,
291
                                 old ? (void __user *)&s2 : NULL);
292
        set_fs(old_fs);
293
        if (ret) return ret;
294
        if (old) {
295
                if (mapout (&s2.sa_mask, tmp2)) return -EINVAL;
296
                tmp = 0; tmp2[2] = 0; tmp2[3] = 0;
297
                if (s2.sa_flags & SA_STACK) tmp |= SOLARIS_SA_ONSTACK;
298
                if (s2.sa_flags & SA_RESTART) tmp |= SOLARIS_SA_RESTART;
299
                if (s2.sa_flags & SA_NOMASK) tmp |= SOLARIS_SA_NODEFER;
300
                if (s2.sa_flags & SA_ONESHOT) tmp |= SOLARIS_SA_RESETHAND;
301
                if (s2.sa_flags & SA_NOCLDSTOP) tmp |= SOLARIS_SA_NOCLDSTOP;
302
                if (put_user (tmp, &p->sa_flags) ||
303
                    __put_user ((u32)(unsigned long)s2.sa_handler, &p->sa_handler) ||
304
                    copy_to_user (&p->sa_mask, tmp2, 4*sizeof(u32)))
305
                        return -EFAULT;
306
        }
307
        return 0;
308
}
309
 
310
asmlinkage int solaris_sigpending(int which, u32 set)
311
{
312
        sigset_t s;
313
        u32 tmp[4];
314
        switch (which) {
315
        case 1: /* sigpending */
316
                spin_lock_irq(&current->sighand->siglock);
317
                sigandsets(&s, &current->blocked, &current->pending.signal);
318
                recalc_sigpending();
319
                spin_unlock_irq(&current->sighand->siglock);
320
                break;
321
        case 2: /* sigfillset - I just set signals which have linux equivalents */
322
                sigfillset(&s);
323
                break;
324
        default: return -EINVAL;
325
        }
326
        if (mapout (&s, tmp)) return -EINVAL;
327
        tmp[2] = 0; tmp[3] = 0;
328
        if (copy_to_user ((u32 __user *)A(set), tmp, sizeof(tmp)))
329
                return -EFAULT;
330
        return 0;
331
}
332
 
333
asmlinkage int solaris_wait(u32 stat_loc)
334
{
335
        unsigned __user *p = (unsigned __user *)A(stat_loc);
336
        int (*sys_wait4)(pid_t,unsigned __user *, int, struct rusage __user *) =
337
                (int (*)(pid_t,unsigned __user *, int, struct rusage __user *))SYS(wait4);
338
        int ret, status;
339
 
340
        ret = sys_wait4(-1, p, WUNTRACED, NULL);
341
        if (ret >= 0 && stat_loc) {
342
                if (get_user (status, p))
343
                        return -EFAULT;
344
                if (((status - 1) & 0xffff) < 0xff)
345
                        status = linux_to_solaris_signals[status & 0x7f] & 0x7f;
346
                else if ((status & 0xff) == 0x7f)
347
                        status = (linux_to_solaris_signals[(status >> 8) & 0xff] << 8) | 0x7f;
348
                if (__put_user (status, p))
349
                        return -EFAULT;
350
        }
351
        return ret;
352
}
353
 
354
asmlinkage int solaris_waitid(int idtype, s32 pid, u32 info, int options)
355
{
356
        int (*sys_wait4)(pid_t,unsigned __user *, int, struct rusage __user *) =
357
                (int (*)(pid_t,unsigned __user *, int, struct rusage __user *))SYS(wait4);
358
        int opts, status, ret;
359
 
360
        switch (idtype) {
361
        case 0: /* P_PID */ break;
362
        case 1: /* P_PGID */ pid = -pid; break;
363
        case 7: /* P_ALL */ pid = -1; break;
364
        default: return -EINVAL;
365
        }
366
        opts = 0;
367
        if (options & SOLARIS_WUNTRACED) opts |= WUNTRACED;
368
        if (options & SOLARIS_WNOHANG) opts |= WNOHANG;
369
        current->state = TASK_RUNNING;
370
        ret = sys_wait4(pid, (unsigned int __user *)A(info), opts, NULL);
371
        if (ret < 0) return ret;
372
        if (info) {
373
                struct sol_siginfo __user *s = (void __user *)A(info);
374
 
375
                if (get_user (status, (unsigned int __user *)A(info)))
376
                        return -EFAULT;
377
 
378
                if (__put_user (SOLARIS_SIGCLD, &s->si_signo) ||
379
                    __put_user (ret, &s->_data._proc._pid))
380
                        return -EFAULT;
381
 
382
                switch (status & 0xff) {
383
                case 0: ret = SOLARIS_CLD_EXITED;
384
                        status = (status >> 8) & 0xff;
385
                        break;
386
                case 0x7f:
387
                        status = (status >> 8) & 0xff;
388
                        switch (status) {
389
                        case SIGSTOP:
390
                        case SIGTSTP: ret = SOLARIS_CLD_STOPPED;
391
                        default: ret = SOLARIS_CLD_EXITED;
392
                        }
393
                        status = linux_to_solaris_signals[status];
394
                        break;
395
                default:
396
                        if (status & 0x80) ret = SOLARIS_CLD_DUMPED;
397
                        else ret = SOLARIS_CLD_KILLED;
398
                        status = linux_to_solaris_signals[status & 0x7f];
399
                        break;
400
                }
401
 
402
                if (__put_user (ret, &s->si_code) ||
403
                    __put_user (status, &s->_data._proc._pdata._cld._status))
404
                        return -EFAULT;
405
        }
406
        return 0;
407
}
408
 
409
extern int svr4_setcontext(svr4_ucontext_t *c, struct pt_regs *regs);
410
extern int svr4_getcontext(svr4_ucontext_t *c, struct pt_regs *regs);
411
 
412
asmlinkage int solaris_context(struct pt_regs *regs)
413
{
414
        switch ((unsigned)regs->u_regs[UREG_I0]) {
415
        case 0: /* getcontext */
416
                return svr4_getcontext((svr4_ucontext_t *)(long)(u32)regs->u_regs[UREG_I1], regs);
417
        case 1: /* setcontext */
418
                return svr4_setcontext((svr4_ucontext_t *)(long)(u32)regs->u_regs[UREG_I1], regs);
419
        default:
420
                return -EINVAL;
421
 
422
        }
423
}
424
 
425
asmlinkage int solaris_sigaltstack(u32 ss, u32 oss)
426
{
427
/* XXX Implement this soon */
428
        return 0;
429
}

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.