blob: 660bc9c89d131941f043cfa061f57ec335bfaaa0 [file] [log] [blame]
Ulrich Weigand65bc8101999-11-12 01:00:34 +00001/*
2 * Sparc signal handling routines
Vincent BĂ©ron9a624912002-05-31 23:06:46 +00003 *
Ulrich Weigand65bc8101999-11-12 01:00:34 +00004 * Copyright 1999 Ulrich Weigand
Alexandre Julliard0799c1a2002-03-09 23:29:33 +00005 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
Jonathan Ernst360a3f92006-05-18 14:49:52 +020018 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Ulrich Weigand65bc8101999-11-12 01:00:34 +000019 */
20
21#ifdef __sparc__
22
23#include "config.h"
Alexandre Julliardfb9cead2005-09-14 10:36:58 +000024#include "wine/port.h"
Ulrich Weigand65bc8101999-11-12 01:00:34 +000025
Alexandre Julliardfb9cead2005-09-14 10:36:58 +000026#include <assert.h>
Ulrich Weigand65bc8101999-11-12 01:00:34 +000027#include <signal.h>
28#include <stdlib.h>
Patrik Stridvalld016f812002-08-17 00:43:16 +000029#ifdef HAVE_UNISTD_H
30# include <unistd.h>
31#endif
Alexandre Julliarde37c6e12003-09-05 23:08:26 +000032#include <stdarg.h>
Ulrich Weigandafd6a4b2000-06-04 01:48:05 +000033#include <stdio.h>
Ulrich Weigand65bc8101999-11-12 01:00:34 +000034#include <sys/ucontext.h>
35
Alexandre Julliard5316dd02009-04-08 19:38:02 +020036#include "ntstatus.h"
37#define WIN32_NO_STATUS
Alexandre Julliarde37c6e12003-09-05 23:08:26 +000038#include "windef.h"
Alexandre Julliarde37c6e12003-09-05 23:08:26 +000039#include "winternl.h"
Patrik Stridvallbc38d6b2001-07-20 18:00:00 +000040#include "winnt.h"
41
42#include "wine/exception.h"
Alexandre Julliard217fdab2003-06-30 21:00:15 +000043#include "ntdll_misc.h"
Patrik Stridvallbc38d6b2001-07-20 18:00:00 +000044
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000045#include "wine/debug.h"
Ulrich Weigand65bc8101999-11-12 01:00:34 +000046
Alexandre Julliard0799c1a2002-03-09 23:29:33 +000047WINE_DEFAULT_DEBUG_CHANNEL(seh);
Ulrich Weigand65bc8101999-11-12 01:00:34 +000048
Alexandre Julliard66255772009-02-18 13:04:50 +010049static pthread_key_t teb_key;
50
Patrik Stridvall43255542002-08-09 01:07:29 +000051typedef int (*wine_signal_handler)(unsigned int sig);
Eric Pouech3d4d7e02002-07-31 18:46:09 +000052
53static wine_signal_handler handlers[256];
54
Eric Pouech3d4d7e02002-07-31 18:46:09 +000055/***********************************************************************
56 * dispatch_signal
57 */
Andrew Talbot01086672007-03-17 10:28:32 +000058static inline int dispatch_signal(unsigned int sig)
Eric Pouech3d4d7e02002-07-31 18:46:09 +000059{
60 if (handlers[sig] == NULL) return 0;
61 return handlers[sig](sig);
62}
63
64
Ulrich Weigand65bc8101999-11-12 01:00:34 +000065/*
66 * FIXME: All this works only on Solaris for now
67 */
68
69/**********************************************************************
70 * save_context
71 */
72static void save_context( CONTEXT *context, ucontext_t *ucontext )
73{
74 /* Special registers */
75 context->psr = ucontext->uc_mcontext.gregs[REG_PSR];
76 context->pc = ucontext->uc_mcontext.gregs[REG_PC];
77 context->npc = ucontext->uc_mcontext.gregs[REG_nPC];
78 context->y = ucontext->uc_mcontext.gregs[REG_Y];
79 context->wim = 0; /* FIXME */
80 context->tbr = 0; /* FIXME */
81
82 /* Global registers */
83 context->g0 = 0; /* always */
84 context->g1 = ucontext->uc_mcontext.gregs[REG_G1];
85 context->g2 = ucontext->uc_mcontext.gregs[REG_G2];
86 context->g3 = ucontext->uc_mcontext.gregs[REG_G3];
87 context->g4 = ucontext->uc_mcontext.gregs[REG_G4];
88 context->g5 = ucontext->uc_mcontext.gregs[REG_G5];
89 context->g6 = ucontext->uc_mcontext.gregs[REG_G6];
90 context->g7 = ucontext->uc_mcontext.gregs[REG_G7];
91
92 /* Current 'out' registers */
93 context->o0 = ucontext->uc_mcontext.gregs[REG_O0];
94 context->o1 = ucontext->uc_mcontext.gregs[REG_O1];
95 context->o2 = ucontext->uc_mcontext.gregs[REG_O2];
96 context->o3 = ucontext->uc_mcontext.gregs[REG_O3];
97 context->o4 = ucontext->uc_mcontext.gregs[REG_O4];
98 context->o5 = ucontext->uc_mcontext.gregs[REG_O5];
99 context->o6 = ucontext->uc_mcontext.gregs[REG_O6];
100 context->o7 = ucontext->uc_mcontext.gregs[REG_O7];
101
102 /* FIXME: what if the current register window isn't saved? */
103 if ( ucontext->uc_mcontext.gwins && ucontext->uc_mcontext.gwins->wbcnt > 0 )
104 {
105 /* Current 'local' registers from first register window */
106 context->l0 = ucontext->uc_mcontext.gwins->wbuf[0].rw_local[0];
107 context->l1 = ucontext->uc_mcontext.gwins->wbuf[0].rw_local[1];
108 context->l2 = ucontext->uc_mcontext.gwins->wbuf[0].rw_local[2];
109 context->l3 = ucontext->uc_mcontext.gwins->wbuf[0].rw_local[3];
110 context->l4 = ucontext->uc_mcontext.gwins->wbuf[0].rw_local[4];
111 context->l5 = ucontext->uc_mcontext.gwins->wbuf[0].rw_local[5];
112 context->l6 = ucontext->uc_mcontext.gwins->wbuf[0].rw_local[6];
113 context->l7 = ucontext->uc_mcontext.gwins->wbuf[0].rw_local[7];
114
115 /* Current 'in' registers from first register window */
116 context->i0 = ucontext->uc_mcontext.gwins->wbuf[0].rw_in[0];
117 context->i1 = ucontext->uc_mcontext.gwins->wbuf[0].rw_in[1];
118 context->i2 = ucontext->uc_mcontext.gwins->wbuf[0].rw_in[2];
119 context->i3 = ucontext->uc_mcontext.gwins->wbuf[0].rw_in[3];
120 context->i4 = ucontext->uc_mcontext.gwins->wbuf[0].rw_in[4];
121 context->i5 = ucontext->uc_mcontext.gwins->wbuf[0].rw_in[5];
122 context->i6 = ucontext->uc_mcontext.gwins->wbuf[0].rw_in[6];
123 context->i7 = ucontext->uc_mcontext.gwins->wbuf[0].rw_in[7];
124 }
125}
126
127/**********************************************************************
128 * restore_context
129 */
130static void restore_context( CONTEXT *context, ucontext_t *ucontext )
131{
132 /* FIXME */
133}
134
135/**********************************************************************
136 * save_fpu
137 */
138static void save_fpu( CONTEXT *context, ucontext_t *ucontext )
139{
140 /* FIXME */
141}
142
143/**********************************************************************
144 * restore_fpu
145 */
146static void restore_fpu( CONTEXT *context, ucontext_t *ucontext )
147{
148 /* FIXME */
149}
150
151
Alexandre Julliardc5a57e72009-04-30 13:08:32 +0200152/**********************************************************************
153 * call_stack_handlers
154 *
155 * Call the stack handlers chain.
156 */
157static NTSTATUS call_stack_handlers( EXCEPTION_RECORD *rec, CONTEXT *context )
158{
159 EXCEPTION_POINTERS ptrs;
160
161 FIXME( "not implemented on Sparc\n" );
162
163 /* hack: call unhandled exception filter directly */
164 ptrs.ExceptionRecord = rec;
165 ptrs.ContextRecord = context;
166 unhandled_exception_filter( &ptrs );
167 return STATUS_UNHANDLED_EXCEPTION;
168}
169
170
171/*******************************************************************
172 * raise_exception
173 *
174 * Implementation of NtRaiseException.
175 */
176static NTSTATUS raise_exception( EXCEPTION_RECORD *rec, CONTEXT *context, BOOL first_chance )
177{
178 NTSTATUS status;
179
180 if (first_chance)
181 {
182 DWORD c;
183
184 TRACE( "code=%x flags=%x addr=%p ip=%x tid=%04x\n",
185 rec->ExceptionCode, rec->ExceptionFlags, rec->ExceptionAddress,
186 context->pc, GetCurrentThreadId() );
187 for (c = 0; c < rec->NumberParameters; c++)
188 TRACE( " info[%d]=%08lx\n", c, rec->ExceptionInformation[c] );
189 if (rec->ExceptionCode == EXCEPTION_WINE_STUB)
190 {
191 if (rec->ExceptionInformation[1] >> 16)
192 MESSAGE( "wine: Call from %p to unimplemented function %s.%s, aborting\n",
193 rec->ExceptionAddress,
194 (char*)rec->ExceptionInformation[0], (char*)rec->ExceptionInformation[1] );
195 else
196 MESSAGE( "wine: Call from %p to unimplemented function %s.%ld, aborting\n",
197 rec->ExceptionAddress,
198 (char*)rec->ExceptionInformation[0], rec->ExceptionInformation[1] );
199 }
200 else
201 {
202 /* FIXME: dump context */
203 }
204
205 status = send_debug_event( rec, TRUE, context );
206 if (status == DBG_CONTINUE || status == DBG_EXCEPTION_HANDLED)
207 return STATUS_SUCCESS;
208
209 if (call_vectored_handlers( rec, context ) == EXCEPTION_CONTINUE_EXECUTION)
210 return STATUS_SUCCESS;
211
212 if ((status = call_stack_handlers( rec, context )) != STATUS_UNHANDLED_EXCEPTION)
213 return status;
214 }
215
216 /* last chance exception */
217
218 status = send_debug_event( rec, FALSE, context );
219 if (status != DBG_CONTINUE)
220 {
221 if (rec->ExceptionFlags & EH_STACK_INVALID)
222 ERR("Exception frame is not in stack limits => unable to dispatch exception.\n");
223 else if (rec->ExceptionCode == STATUS_NONCONTINUABLE_EXCEPTION)
224 ERR("Process attempted to continue execution after noncontinuable exception.\n");
225 else
226 ERR("Unhandled exception code %x flags %x addr %p\n",
227 rec->ExceptionCode, rec->ExceptionFlags, rec->ExceptionAddress );
Alexandre Julliard44ed5c02009-10-13 14:15:27 +0200228 NtTerminateProcess( NtCurrentProcess(), rec->ExceptionCode );
Alexandre Julliardc5a57e72009-04-30 13:08:32 +0200229 }
230 return STATUS_SUCCESS;
231}
232
233
Alexandre Julliard2654be02006-01-11 20:20:32 +0100234/***********************************************************************
Alexandre Julliard14c452f2009-01-06 17:50:34 +0100235 * RtlCaptureContext (NTDLL.@)
Alexandre Julliardd2ad6f82006-04-20 15:40:28 +0200236 */
Alexandre Julliard14c452f2009-01-06 17:50:34 +0100237void WINAPI RtlCaptureContext( CONTEXT *context )
Alexandre Julliardd2ad6f82006-04-20 15:40:28 +0200238{
239 FIXME("not implemented\n");
Alexandre Julliard14c452f2009-01-06 17:50:34 +0100240 memset( context, 0, sizeof(*context) );
Alexandre Julliardd2ad6f82006-04-20 15:40:28 +0200241}
242
243
244/***********************************************************************
Alexandre Julliard2654be02006-01-11 20:20:32 +0100245 * set_cpu_context
246 *
247 * Set the new CPU context.
248 */
249void set_cpu_context( const CONTEXT *context )
250{
251 FIXME("not implemented\n");
252}
253
254
Alexandre Julliard75be87d2009-03-13 11:31:25 +0100255/***********************************************************************
256 * copy_context
257 *
258 * Copy a register context according to the flags.
259 */
260void copy_context( CONTEXT *to, const CONTEXT *from, DWORD flags )
261{
262 flags &= ~CONTEXT_SPARC; /* get rid of CPU id */
263 if (flags & CONTEXT_CONTROL)
264 {
265 to->psr = from->psr;
266 to->pc = from->pc;
267 to->npc = from->npc;
268 to->y = from->y;
269 to->wim = from->wim;
270 to->tbr = from->tbr;
271 }
272 if (flags & CONTEXT_INTEGER)
273 {
274 to->g0 = from->g0;
275 to->g1 = from->g1;
276 to->g2 = from->g2;
277 to->g3 = from->g3;
278 to->g4 = from->g4;
279 to->g5 = from->g5;
280 to->g6 = from->g6;
281 to->g7 = from->g7;
282 to->o0 = from->o0;
283 to->o1 = from->o1;
284 to->o2 = from->o2;
285 to->o3 = from->o3;
286 to->o4 = from->o4;
287 to->o5 = from->o5;
288 to->o6 = from->o6;
289 to->o7 = from->o7;
290 to->l0 = from->l0;
291 to->l1 = from->l1;
292 to->l2 = from->l2;
293 to->l3 = from->l3;
294 to->l4 = from->l4;
295 to->l5 = from->l5;
296 to->l6 = from->l6;
297 to->l7 = from->l7;
298 to->i0 = from->i0;
299 to->i1 = from->i1;
300 to->i2 = from->i2;
301 to->i3 = from->i3;
302 to->i4 = from->i4;
303 to->i5 = from->i5;
304 to->i6 = from->i6;
305 to->i7 = from->i7;
306 }
307 if (flags & CONTEXT_FLOATING_POINT)
308 {
309 /* FIXME */
310 }
311}
312
313
Alexandre Julliard5316dd02009-04-08 19:38:02 +0200314/***********************************************************************
315 * context_to_server
316 *
317 * Convert a register context to the server format.
318 */
319NTSTATUS context_to_server( context_t *to, const CONTEXT *from )
320{
321 DWORD flags = from->ContextFlags & ~CONTEXT_SPARC; /* get rid of CPU id */
322
323 memset( to, 0, sizeof(*to) );
324 to->cpu = CPU_SPARC;
325
326 if (flags & CONTEXT_CONTROL)
327 {
328 to->flags |= SERVER_CTX_CONTROL;
329 to->ctl.sparc_regs.psr = from->psr;
330 to->ctl.sparc_regs.pc = from->pc;
331 to->ctl.sparc_regs.npc = from->npc;
332 to->ctl.sparc_regs.y = from->y;
333 to->ctl.sparc_regs.wim = from->wim;
334 to->ctl.sparc_regs.tbr = from->tbr;
335 }
336 if (flags & CONTEXT_INTEGER)
337 {
338 to->flags |= SERVER_CTX_INTEGER;
339 to->integer.sparc_regs.g[0] = from->g0;
340 to->integer.sparc_regs.g[1] = from->g1;
341 to->integer.sparc_regs.g[2] = from->g2;
342 to->integer.sparc_regs.g[3] = from->g3;
343 to->integer.sparc_regs.g[4] = from->g4;
344 to->integer.sparc_regs.g[5] = from->g5;
345 to->integer.sparc_regs.g[6] = from->g6;
346 to->integer.sparc_regs.g[7] = from->g7;
347 to->integer.sparc_regs.o[0] = from->o0;
348 to->integer.sparc_regs.o[1] = from->o1;
349 to->integer.sparc_regs.o[2] = from->o2;
350 to->integer.sparc_regs.o[3] = from->o3;
351 to->integer.sparc_regs.o[4] = from->o4;
352 to->integer.sparc_regs.o[5] = from->o5;
353 to->integer.sparc_regs.o[6] = from->o6;
354 to->integer.sparc_regs.o[7] = from->o7;
355 to->integer.sparc_regs.l[0] = from->l0;
356 to->integer.sparc_regs.l[1] = from->l1;
357 to->integer.sparc_regs.l[2] = from->l2;
358 to->integer.sparc_regs.l[3] = from->l3;
359 to->integer.sparc_regs.l[4] = from->l4;
360 to->integer.sparc_regs.l[5] = from->l5;
361 to->integer.sparc_regs.l[6] = from->l6;
362 to->integer.sparc_regs.l[7] = from->l7;
363 to->integer.sparc_regs.i[0] = from->i0;
364 to->integer.sparc_regs.i[1] = from->i1;
365 to->integer.sparc_regs.i[2] = from->i2;
366 to->integer.sparc_regs.i[3] = from->i3;
367 to->integer.sparc_regs.i[4] = from->i4;
368 to->integer.sparc_regs.i[5] = from->i5;
369 to->integer.sparc_regs.i[6] = from->i6;
370 to->integer.sparc_regs.i[7] = from->i7;
371 }
372 if (flags & CONTEXT_FLOATING_POINT)
373 {
374 /* FIXME */
375 }
376 return STATUS_SUCCESS;
377}
378
379
380/***********************************************************************
381 * context_from_server
382 *
383 * Convert a register context from the server format.
384 */
385NTSTATUS context_from_server( CONTEXT *to, const context_t *from )
386{
387 if (from->cpu != CPU_SPARC) return STATUS_INVALID_PARAMETER;
388
389 to->ContextFlags = CONTEXT_SPARC;
390 if (from->flags & SERVER_CTX_CONTROL)
391 {
392 to->ContextFlags |= CONTEXT_CONTROL;
393 to->psr = from->ctl.sparc_regs.psr;
394 to->pc = from->ctl.sparc_regs.pc;
395 to->npc = from->ctl.sparc_regs.npc;
396 to->y = from->ctl.sparc_regs.y;
397 to->wim = from->ctl.sparc_regs.wim;
398 to->tbr = from->ctl.sparc_regs.tbr;
399 }
400 if (from->flags & SERVER_CTX_INTEGER)
401 {
402 to->ContextFlags |= CONTEXT_INTEGER;
403 to->g0 = from->integer.sparc_regs.g[0];
404 to->g1 = from->integer.sparc_regs.g[1];
405 to->g2 = from->integer.sparc_regs.g[2];
406 to->g3 = from->integer.sparc_regs.g[3];
407 to->g4 = from->integer.sparc_regs.g[4];
408 to->g5 = from->integer.sparc_regs.g[5];
409 to->g6 = from->integer.sparc_regs.g[6];
410 to->g7 = from->integer.sparc_regs.g[7];
411 to->o0 = from->integer.sparc_regs.o[0];
412 to->o1 = from->integer.sparc_regs.o[1];
413 to->o2 = from->integer.sparc_regs.o[2];
414 to->o3 = from->integer.sparc_regs.o[3];
415 to->o4 = from->integer.sparc_regs.o[4];
416 to->o5 = from->integer.sparc_regs.o[5];
417 to->o6 = from->integer.sparc_regs.o[6];
418 to->o7 = from->integer.sparc_regs.o[7];
419 to->l0 = from->integer.sparc_regs.l[0];
420 to->l1 = from->integer.sparc_regs.l[1];
421 to->l2 = from->integer.sparc_regs.l[2];
422 to->l3 = from->integer.sparc_regs.l[3];
423 to->l4 = from->integer.sparc_regs.l[4];
424 to->l5 = from->integer.sparc_regs.l[5];
425 to->l6 = from->integer.sparc_regs.l[6];
426 to->l7 = from->integer.sparc_regs.l[7];
427 to->i0 = from->integer.sparc_regs.i[0];
428 to->i1 = from->integer.sparc_regs.i[1];
429 to->i2 = from->integer.sparc_regs.i[2];
430 to->i3 = from->integer.sparc_regs.i[3];
431 to->i4 = from->integer.sparc_regs.i[4];
432 to->i5 = from->integer.sparc_regs.i[5];
433 to->i6 = from->integer.sparc_regs.i[6];
434 to->i7 = from->integer.sparc_regs.i[7];
435 }
436 if (from->flags & SERVER_CTX_FLOATING_POINT)
437 {
438 /* FIXME */
439 }
440 return STATUS_SUCCESS;
441}
442
443
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000444/**********************************************************************
445 * segv_handler
446 *
447 * Handler for SIGSEGV.
448 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200449static void segv_handler( int signal, siginfo_t *info, void *ucontext )
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000450{
451 EXCEPTION_RECORD rec;
452 CONTEXT context;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200453 NTSTATUS status;
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000454
Alexandre Julliard81b9ca52008-11-25 12:02:16 +0100455 rec.ExceptionCode = EXCEPTION_ACCESS_VIOLATION;
456
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000457 /* we want the page-fault case to be fast */
458 if ( info->si_code == SEGV_ACCERR )
Alexandre Julliard81b9ca52008-11-25 12:02:16 +0100459 if (!(rec.ExceptionCode = virtual_handle_fault( info->si_addr, 0 ))) return;
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000460
461 save_context( &context, ucontext );
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000462 rec.ExceptionRecord = NULL;
463 rec.ExceptionFlags = EXCEPTION_CONTINUABLE;
464 rec.ExceptionAddress = (LPVOID)context.pc;
465 rec.NumberParameters = 2;
466 rec.ExceptionInformation[0] = 0; /* FIXME: read/write access ? */
Dmitry Timoshkov0497af02005-02-24 13:15:36 +0000467 rec.ExceptionInformation[1] = (ULONG_PTR)info->si_addr;
Vincent BĂ©ron9a624912002-05-31 23:06:46 +0000468
Alexandre Julliard48199d72009-04-10 13:09:06 +0200469 status = raise_exception( &rec, &context, TRUE );
470 if (status) raise_status( status, &rec );
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000471 restore_context( &context, ucontext );
472}
473
474/**********************************************************************
475 * bus_handler
476 *
477 * Handler for SIGBUS.
478 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200479static void bus_handler( int signal, siginfo_t *info, void *ucontext )
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000480{
481 EXCEPTION_RECORD rec;
482 CONTEXT context;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200483 NTSTATUS status;
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000484
485 save_context( &context, ucontext );
486 rec.ExceptionRecord = NULL;
487 rec.ExceptionFlags = EXCEPTION_CONTINUABLE;
488 rec.ExceptionAddress = (LPVOID)context.pc;
489 rec.NumberParameters = 0;
490
491 if ( info->si_code == BUS_ADRALN )
492 rec.ExceptionCode = EXCEPTION_DATATYPE_MISALIGNMENT;
493 else
494 rec.ExceptionCode = EXCEPTION_ACCESS_VIOLATION;
Vincent BĂ©ron9a624912002-05-31 23:06:46 +0000495
Alexandre Julliard48199d72009-04-10 13:09:06 +0200496 status = raise_exception( &rec, &context, TRUE );
497 if (status) raise_status( status, &rec );
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000498 restore_context( &context, ucontext );
499}
500
501/**********************************************************************
502 * ill_handler
503 *
504 * Handler for SIGILL.
505 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200506static void ill_handler( int signal, siginfo_t *info, void *ucontext )
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000507{
508 EXCEPTION_RECORD rec;
509 CONTEXT context;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200510 NTSTATUS status;
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000511
512 switch ( info->si_code )
513 {
514 default:
515 case ILL_ILLOPC:
516 case ILL_ILLOPN:
517 case ILL_ILLADR:
518 case ILL_ILLTRP:
519 rec.ExceptionCode = EXCEPTION_ILLEGAL_INSTRUCTION;
520 break;
521
522 case ILL_PRVOPC:
523 case ILL_PRVREG:
524 rec.ExceptionCode = EXCEPTION_PRIV_INSTRUCTION;
525 break;
526
527 case ILL_BADSTK:
528 rec.ExceptionCode = EXCEPTION_STACK_OVERFLOW;
529 break;
530 }
Vincent BĂ©ron9a624912002-05-31 23:06:46 +0000531
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000532 save_context( &context, ucontext );
533 rec.ExceptionRecord = NULL;
534 rec.ExceptionFlags = EXCEPTION_CONTINUABLE;
535 rec.ExceptionAddress = (LPVOID)context.pc;
536 rec.NumberParameters = 0;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200537 status = raise_exception( &rec, &context, TRUE );
538 if (status) raise_status( status, &rec );
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000539 restore_context( &context, ucontext );
540}
541
542
543/**********************************************************************
544 * trap_handler
545 *
546 * Handler for SIGTRAP.
547 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200548static void trap_handler( int signal, siginfo_t *info, void *ucontext )
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000549{
550 EXCEPTION_RECORD rec;
551 CONTEXT context;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200552 NTSTATUS status;
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000553
554 switch ( info->si_code )
555 {
556 case TRAP_TRACE:
557 rec.ExceptionCode = EXCEPTION_SINGLE_STEP;
558 break;
559 case TRAP_BRKPT:
560 default:
561 rec.ExceptionCode = EXCEPTION_BREAKPOINT;
562 break;
563 }
564
565 save_context( &context, ucontext );
566 rec.ExceptionFlags = EXCEPTION_CONTINUABLE;
567 rec.ExceptionRecord = NULL;
568 rec.ExceptionAddress = (LPVOID)context.pc;
569 rec.NumberParameters = 0;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200570 status = raise_exception( &rec, &context, TRUE );
571 if (status) raise_status( status, &rec );
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000572 restore_context( &context, ucontext );
573}
574
575
576/**********************************************************************
577 * fpe_handler
578 *
579 * Handler for SIGFPE.
580 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200581static void fpe_handler( int signal, siginfo_t *info, void *ucontext )
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000582{
583 EXCEPTION_RECORD rec;
584 CONTEXT context;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200585 NTSTATUS status;
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000586
587 switch ( info->si_code )
588 {
589 case FPE_FLTSUB:
590 rec.ExceptionCode = EXCEPTION_ARRAY_BOUNDS_EXCEEDED;
591 break;
592 case FPE_INTDIV:
593 rec.ExceptionCode = EXCEPTION_INT_DIVIDE_BY_ZERO;
594 break;
595 case FPE_INTOVF:
596 rec.ExceptionCode = EXCEPTION_INT_OVERFLOW;
597 break;
598 case FPE_FLTDIV:
599 rec.ExceptionCode = EXCEPTION_FLT_DIVIDE_BY_ZERO;
600 break;
601 case FPE_FLTOVF:
602 rec.ExceptionCode = EXCEPTION_FLT_OVERFLOW;
603 break;
604 case FPE_FLTUND:
605 rec.ExceptionCode = EXCEPTION_FLT_UNDERFLOW;
606 break;
607 case FPE_FLTRES:
608 rec.ExceptionCode = EXCEPTION_FLT_INEXACT_RESULT;
609 break;
610 case FPE_FLTINV:
611 default:
612 rec.ExceptionCode = EXCEPTION_FLT_INVALID_OPERATION;
613 break;
614 }
615
616 save_context( &context, ucontext );
617 save_fpu( &context, ucontext );
618 rec.ExceptionFlags = EXCEPTION_CONTINUABLE;
619 rec.ExceptionRecord = NULL;
620 rec.ExceptionAddress = (LPVOID)context.pc;
621 rec.NumberParameters = 0;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200622 status = raise_exception( &rec, &context, TRUE );
623 if (status) raise_status( status, &rec );
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000624 restore_context( &context, ucontext );
625 restore_fpu( &context, ucontext );
626}
627
628
629/**********************************************************************
630 * int_handler
631 *
632 * Handler for SIGINT.
633 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200634static void int_handler( int signal, siginfo_t *info, void *ucontext )
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000635{
Eric Pouech3d4d7e02002-07-31 18:46:09 +0000636 if (!dispatch_signal(SIGINT))
Eric Pouech93bfa0d2002-06-02 21:22:22 +0000637 {
638 EXCEPTION_RECORD rec;
639 CONTEXT context;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200640 NTSTATUS status;
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000641
Eric Pouech93bfa0d2002-06-02 21:22:22 +0000642 save_context( &context, ucontext );
643 rec.ExceptionCode = CONTROL_C_EXIT;
644 rec.ExceptionFlags = EXCEPTION_CONTINUABLE;
645 rec.ExceptionRecord = NULL;
646 rec.ExceptionAddress = (LPVOID)context.pc;
647 rec.NumberParameters = 0;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200648 status = raise_exception( &rec, &context, TRUE );
649 if (status) raise_status( status, &rec );
Eric Pouech93bfa0d2002-06-02 21:22:22 +0000650 restore_context( &context, ucontext );
651 }
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000652}
653
Lionel Ulmerbdb44552002-10-28 23:56:58 +0000654/**********************************************************************
655 * abrt_handler
656 *
657 * Handler for SIGABRT.
658 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200659static void abrt_handler( int signal, struct siginfo *info, void *ucontext )
Lionel Ulmerbdb44552002-10-28 23:56:58 +0000660{
661 EXCEPTION_RECORD rec;
662 CONTEXT context;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200663 NTSTATUS status;
Lionel Ulmerbdb44552002-10-28 23:56:58 +0000664
Alexandre Julliard9968d842009-05-01 17:57:29 +0200665 save_context( &context, ucontext );
Lionel Ulmerbdb44552002-10-28 23:56:58 +0000666 rec.ExceptionCode = EXCEPTION_WINE_ASSERTION;
667 rec.ExceptionFlags = EH_NONCONTINUABLE;
668 rec.ExceptionRecord = NULL;
Juraj Hercek386d51d2002-11-08 18:54:10 +0000669 rec.ExceptionAddress = (LPVOID)context.pc;
Lionel Ulmerbdb44552002-10-28 23:56:58 +0000670 rec.NumberParameters = 0;
Alexandre Julliard48199d72009-04-10 13:09:06 +0200671 status = raise_exception( &rec, &context, TRUE );
672 if (status) raise_status( status, &rec );
Alexandre Julliard9968d842009-05-01 17:57:29 +0200673 restore_context( &context, ucontext );
Lionel Ulmerbdb44552002-10-28 23:56:58 +0000674}
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000675
Alexandre Julliardd04ccb82003-03-04 22:18:43 +0000676
677/**********************************************************************
Alexandre Julliard3b244b92007-12-20 12:19:35 +0100678 * quit_handler
Alexandre Julliard6a26e3f2003-03-21 23:45:26 +0000679 *
Alexandre Julliard3b244b92007-12-20 12:19:35 +0100680 * Handler for SIGQUIT.
Alexandre Julliard6a26e3f2003-03-21 23:45:26 +0000681 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200682static void quit_handler( int signal, struct siginfo *info, void *ucontext )
Alexandre Julliard6a26e3f2003-03-21 23:45:26 +0000683{
Alexandre Julliardcc933f52009-02-20 11:23:08 +0100684 abort_thread(0);
Alexandre Julliard6a26e3f2003-03-21 23:45:26 +0000685}
686
687
688/**********************************************************************
Alexandre Julliardd04ccb82003-03-04 22:18:43 +0000689 * usr1_handler
690 *
691 * Handler for SIGUSR1, used to signal a thread that it got suspended.
692 */
Alexandre Julliard9968d842009-05-01 17:57:29 +0200693static void usr1_handler( int signal, struct siginfo *info, void *ucontext )
Alexandre Julliardd04ccb82003-03-04 22:18:43 +0000694{
Alexandre Julliard73c72392005-11-02 20:54:12 +0000695 CONTEXT context;
Eric Pouech14d04b62003-04-03 23:57:11 +0000696
Alexandre Julliard9968d842009-05-01 17:57:29 +0200697 save_context( &context, ucontext );
Alexandre Julliard73c72392005-11-02 20:54:12 +0000698 wait_suspend( &context );
Alexandre Julliard9968d842009-05-01 17:57:29 +0200699 restore_context( &context, ucontext );
Alexandre Julliardfb9cead2005-09-14 10:36:58 +0000700}
701
702
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000703/***********************************************************************
Eric Pouech3d4d7e02002-07-31 18:46:09 +0000704 * __wine_set_signal_handler (NTDLL.@)
705 */
Maarten Lankhorst768160e2008-12-16 16:37:46 +0100706int CDECL __wine_set_signal_handler(unsigned int sig, wine_signal_handler wsh)
Eric Pouech3d4d7e02002-07-31 18:46:09 +0000707{
708 if (sig > sizeof(handlers) / sizeof(handlers[0])) return -1;
709 if (handlers[sig] != NULL) return -2;
710 handlers[sig] = wsh;
711 return 0;
712}
713
714
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000715/**********************************************************************
Alexandre Julliardd4f1fff2009-11-19 12:25:52 +0100716 * signal_alloc_thread
717 */
718NTSTATUS signal_alloc_thread( TEB **teb )
719{
720 static size_t sigstack_zero_bits;
721 SIZE_T size;
722 NTSTATUS status;
723
724 if (!sigstack_zero_bits)
725 {
726 size_t min_size = getpagesize(); /* this is just for the TEB, we don't use a signal stack yet */
727 /* find the first power of two not smaller than min_size */
728 while ((1u << sigstack_zero_bits) < min_size) sigstack_zero_bits++;
729 assert( sizeof(TEB) <= min_size );
730 }
731
732 size = 1 << sigstack_zero_bits;
733 *teb = NULL;
734 if (!(status = NtAllocateVirtualMemory( NtCurrentProcess(), (void **)teb, sigstack_zero_bits,
735 &size, MEM_COMMIT | MEM_TOP_DOWN, PAGE_READWRITE )))
736 {
737 (*teb)->Tib.Self = &(*teb)->Tib;
738 (*teb)->Tib.ExceptionList = (void *)~0UL;
739 }
740 return status;
741}
742
743
744/**********************************************************************
745 * signal_free_thread
746 */
747void signal_free_thread( TEB *teb )
748{
749 SIZE_T size;
750
751 if (teb->DeallocationStack)
752 {
753 size = 0;
754 NtFreeVirtualMemory( GetCurrentProcess(), &teb->DeallocationStack, &size, MEM_RELEASE );
755 }
756 size = 0;
757 NtFreeVirtualMemory( NtCurrentProcess(), (void **)&teb, &size, MEM_RELEASE );
758}
759
760
761/**********************************************************************
Alexandre Julliard531ff0b2008-07-03 20:18:23 +0200762 * signal_init_thread
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000763 */
Alexandre Julliard66255772009-02-18 13:04:50 +0100764void signal_init_thread( TEB *teb )
Alexandre Julliard531ff0b2008-07-03 20:18:23 +0200765{
Alexandre Julliard66255772009-02-18 13:04:50 +0100766 static int init_done;
767
768 if (!init_done)
769 {
770 pthread_key_create( &teb_key, NULL );
771 init_done = 1;
772 }
773 pthread_setspecific( teb_key, teb );
Alexandre Julliard531ff0b2008-07-03 20:18:23 +0200774}
775
Alexandre Julliard66255772009-02-18 13:04:50 +0100776
Alexandre Julliard531ff0b2008-07-03 20:18:23 +0200777/**********************************************************************
778 * signal_init_process
779 */
780void signal_init_process(void)
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000781{
Alexandre Julliard9968d842009-05-01 17:57:29 +0200782 struct sigaction sig_act;
783
784 sig_act.sa_mask = server_block_set;
785 sig_act.sa_flags = SA_RESTART | SA_SIGINFO;
786
787 sig_act.sa_sigaction = int_handler;
788 if (sigaction( SIGINT, &sig_act, NULL ) == -1) goto error;
789 sig_act.sa_sigaction = fpe_handler;
790 if (sigaction( SIGFPE, &sig_act, NULL ) == -1) goto error;
791 sig_act.sa_sigaction = abrt_handler;
792 if (sigaction( SIGABRT, &sig_act, NULL ) == -1) goto error;
793 sig_act.sa_sigaction = quit_handler;
794 if (sigaction( SIGQUIT, &sig_act, NULL ) == -1) goto error;
795 sig_act.sa_sigaction = usr1_handler;
796 if (sigaction( SIGUSR1, &sig_act, NULL ) == -1) goto error;
797
798 sig_act.sa_sigaction = segv_handler;
799 if (sigaction( SIGSEGV, &sig_act, NULL ) == -1) goto error;
800 if (sigaction( SIGILL, &sig_act, NULL ) == -1) goto error;
801#ifdef SIGBUS
802 if (sigaction( SIGBUS, &sig_act, NULL ) == -1) goto error;
803#endif
804
805#ifdef SIGTRAP
806 sig_act.sa_sigaction = trap_handler;
807 if (sigaction( SIGTRAP, &sig_act, NULL ) == -1) goto error;
808#endif
809
Eric Friasc8866c02004-11-21 15:38:26 +0000810 /* 'ta 6' tells the kernel to synthesize any unaligned accesses this
811 process makes, instead of just signalling an error and terminating
812 the process. wine-devel did not reach a conclusion on whether
813 this is correct, because that is what x86 does, or it is harmful
814 because it could obscure problems in user code */
Yann Droneaudc8553642009-10-26 15:33:15 +0100815 __asm__("ta 6"); /* 6 == ST_FIX_ALIGN defined in sys/trap.h */
Alexandre Julliard531ff0b2008-07-03 20:18:23 +0200816 return;
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000817
818 error:
819 perror("sigaction");
Alexandre Julliard531ff0b2008-07-03 20:18:23 +0200820 exit(1);
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000821}
822
Alexandre Julliard50c6b742002-01-07 18:02:35 +0000823
824/**********************************************************************
Alexandre Julliardb59627c2000-09-24 03:11:54 +0000825 * __wine_enter_vm86
826 */
827void __wine_enter_vm86( CONTEXT *context )
828{
829 MESSAGE("vm86 mode not supported on this platform\n");
830}
831
Alexandre Julliardf4eee1c2009-04-10 13:14:05 +0200832/***********************************************************************
Alexandre Julliardc5a57e72009-04-30 13:08:32 +0200833 * RtlUnwind (NTDLL.@)
834 */
835void WINAPI RtlUnwind( PVOID pEndFrame, PVOID targetIp, PEXCEPTION_RECORD pRecord, PVOID retval )
836{
837 FIXME( "Not implemented on Sparc\n" );
838}
839
840/*******************************************************************
841 * NtRaiseException (NTDLL.@)
842 */
843NTSTATUS WINAPI NtRaiseException( EXCEPTION_RECORD *rec, CONTEXT *context, BOOL first_chance )
844{
845 NTSTATUS status = raise_exception( rec, context, first_chance );
846 if (status == STATUS_SUCCESS) NtSetContextThread( GetCurrentThread(), context );
847 return status;
848}
849
850/***********************************************************************
Alexandre Julliardf4eee1c2009-04-10 13:14:05 +0200851 * RtlRaiseException (NTDLL.@)
852 */
853void WINAPI RtlRaiseException( EXCEPTION_RECORD *rec )
854{
855 CONTEXT context;
856 NTSTATUS status;
857
858 RtlCaptureContext( &context );
Alexandre Julliard7f975052009-04-10 13:17:12 +0200859 rec->ExceptionAddress = (void *)context.pc;
Alexandre Julliardf4eee1c2009-04-10 13:14:05 +0200860 status = raise_exception( rec, &context, TRUE );
861 if (status) raise_status( status, rec );
862}
863
Alexandre Julliardd45fca82009-11-11 17:19:23 +0100864/*************************************************************************
865 * RtlCaptureStackBackTrace (NTDLL.@)
866 */
867USHORT WINAPI RtlCaptureStackBackTrace( ULONG skip, ULONG count, PVOID *buffer, ULONG *hash )
868{
869 FIXME( "(%d, %d, %p, %p) stub!\n", skip, count, buffer, hash );
870 return 0;
871}
872
Alexandre Julliardf124c7c2009-06-18 16:38:30 +0200873/***********************************************************************
874 * call_thread_entry_point
875 */
876void call_thread_entry_point( LPTHREAD_START_ROUTINE entry, void *arg )
877{
878 __TRY
879 {
Alexandre Julliard06105492009-06-18 16:50:34 +0200880 exit_thread( entry( arg ));
Alexandre Julliardf124c7c2009-06-18 16:38:30 +0200881 }
882 __EXCEPT(unhandled_exception_filter)
883 {
884 NtTerminateThread( GetCurrentThread(), GetExceptionCode() );
885 }
886 __ENDTRY
887 abort(); /* should not be reached */
888}
889
Alexandre Julliard06105492009-06-18 16:50:34 +0200890/***********************************************************************
891 * RtlExitUserThread (NTDLL.@)
892 */
893void WINAPI RtlExitUserThread( ULONG status )
894{
895 exit_thread( status );
896}
897
Alexandre Julliard8101a2f2009-08-28 11:52:21 +0200898/***********************************************************************
899 * abort_thread
900 */
901void abort_thread( int status )
902{
903 terminate_thread( status );
904}
905
Alexandre Julliardb59627c2000-09-24 03:11:54 +0000906/**********************************************************************
Patrik Stridvall3ca98232001-06-20 23:03:14 +0000907 * DbgBreakPoint (NTDLL.@)
Ulrich Weigandafd6a4b2000-06-04 01:48:05 +0000908 */
909void WINAPI DbgBreakPoint(void)
910{
Gregg Mattinson07db3252002-06-21 20:10:07 +0000911 kill(getpid(), SIGTRAP);
Ulrich Weigandafd6a4b2000-06-04 01:48:05 +0000912}
913
914/**********************************************************************
Patrik Stridvall3ca98232001-06-20 23:03:14 +0000915 * DbgUserBreakPoint (NTDLL.@)
Ulrich Weigandafd6a4b2000-06-04 01:48:05 +0000916 */
917void WINAPI DbgUserBreakPoint(void)
918{
Gregg Mattinson07db3252002-06-21 20:10:07 +0000919 kill(getpid(), SIGTRAP);
Ulrich Weigandafd6a4b2000-06-04 01:48:05 +0000920}
921
Alexandre Julliard66255772009-02-18 13:04:50 +0100922/**********************************************************************
923 * NtCurrentTeb (NTDLL.@)
924 */
925TEB * WINAPI NtCurrentTeb(void)
926{
927 return pthread_getspecific( teb_key );
928}
929
Ulrich Weigand65bc8101999-11-12 01:00:34 +0000930#endif /* __sparc__ */