No longer directly accessing debuggee memory.
[wine] / debugger / registers.c
1 /*
2  * Debugger register handling
3  *
4  * Copyright 1995 Alexandre Julliard
5  */
6
7 #include "config.h"
8 #include <stdio.h>
9 #include <string.h>
10 #include "debugger.h"
11
12 /***********************************************************************
13  *           DEBUG_SetRegister
14  *
15  * Set a register value.
16  */
17 void DEBUG_SetRegister( enum debug_regs reg, int val )
18 {
19 #ifdef __i386__
20     switch(reg)
21     {
22         case REG_EAX: DEBUG_context.Eax = val; break;
23         case REG_EBX: DEBUG_context.Ebx = val; break;
24         case REG_ECX: DEBUG_context.Ecx = val; break;
25         case REG_EDX: DEBUG_context.Edx = val; break;
26         case REG_ESI: DEBUG_context.Esi = val; break;
27         case REG_EDI: DEBUG_context.Edi = val; break;
28         case REG_EBP: DEBUG_context.Ebp = val; break;
29         case REG_EFL: DEBUG_context.EFlags = val; break;
30         case REG_EIP: DEBUG_context.Eip = val; break;
31         case REG_ESP: DEBUG_context.Esp = val; break;
32         case REG_CS:  DEBUG_context.SegCs = val; break;
33         case REG_DS:  DEBUG_context.SegDs = val; break;
34         case REG_ES:  DEBUG_context.SegEs = val; break;
35         case REG_SS:  DEBUG_context.SegSs = val; break;
36         case REG_FS:  DEBUG_context.SegFs = val; break;
37         case REG_GS:  DEBUG_context.SegGs = val; break;
38 #define SET_LOW_WORD(dw,lw)     ((dw) = ((dw) & 0xFFFF0000) | LOWORD(lw))
39         case REG_AX:  SET_LOW_WORD(DEBUG_context.Eax,val); break;
40         case REG_BX:  SET_LOW_WORD(DEBUG_context.Ebx,val); break;
41         case REG_CX:  SET_LOW_WORD(DEBUG_context.Ecx,val); break;
42         case REG_DX:  SET_LOW_WORD(DEBUG_context.Edx,val); break;
43         case REG_SI:  SET_LOW_WORD(DEBUG_context.Esi,val); break;
44         case REG_DI:  SET_LOW_WORD(DEBUG_context.Edi,val); break;
45         case REG_BP:  SET_LOW_WORD(DEBUG_context.Ebp,val); break;
46         case REG_FL:  SET_LOW_WORD(DEBUG_context.EFlags,val); break;
47         case REG_IP:  SET_LOW_WORD(DEBUG_context.Eip,val); break;
48         case REG_SP:  SET_LOW_WORD(DEBUG_context.Esp,val); break;
49 #undef SET_LOWORD
50     }
51 #endif
52 }
53
54
55 int DEBUG_PrintRegister(enum debug_regs reg)
56 {
57 #ifdef __i386__
58     char*       val = NULL;
59     switch(reg)
60     {
61         case REG_EAX: val = "%%eax"; break;
62         case REG_EBX: val = "%%ebx"; break;
63         case REG_ECX: val = "%%ecx"; break;
64         case REG_EDX: val = "%%edx"; break;
65         case REG_ESI: val = "%%esi"; break;
66         case REG_EDI: val = "%%edi"; break;
67         case REG_EBP: val = "%%ebp"; break;
68         case REG_EFL: val = "%%efl"; break;
69         case REG_EIP: val = "%%eip"; break;
70         case REG_ESP: val = "%%esp"; break;
71         case REG_AX:  val = "%%ax";  break;
72         case REG_BX:  val = "%%bx";  break;
73         case REG_CX:  val = "%%cx";  break;
74         case REG_DX:  val = "%%dx";  break;
75         case REG_SI:  val = "%%si";  break;
76         case REG_DI:  val = "%%di";  break;
77         case REG_BP:  val = "%%bp";  break;
78         case REG_FL:  val = "%%fl";  break;
79         case REG_IP:  val = "%%ip";  break;
80         case REG_SP:  val = "%%sp";  break;
81         case REG_CS:  val = "%%cs";  break;
82         case REG_DS:  val = "%%ds";  break;
83         case REG_ES:  val = "%%es";  break;
84         case REG_SS:  val = "%%ss";  break;
85         case REG_FS:  val = "%%fs";  break;
86         case REG_GS:  val = "%%gs";  break;
87     }
88     if (val) fprintf(stderr, val);
89     return TRUE;
90 #else
91     return FALSE;
92 #endif
93 }
94
95 /***********************************************************************
96  *           DEBUG_GetRegister
97  *
98  * Get a register value.
99  */
100 int DEBUG_GetRegister( enum debug_regs reg )
101 {
102 #ifdef __i386__
103     switch(reg)
104     {
105         case REG_EAX: return DEBUG_context.Eax;
106         case REG_EBX: return DEBUG_context.Ebx;
107         case REG_ECX: return DEBUG_context.Ecx;
108         case REG_EDX: return DEBUG_context.Edx;
109         case REG_ESI: return DEBUG_context.Esi;
110         case REG_EDI: return DEBUG_context.Edi;
111         case REG_EBP: return DEBUG_context.Ebp;
112         case REG_EFL: return DEBUG_context.EFlags;
113         case REG_EIP: return DEBUG_context.Eip;
114         case REG_ESP: return DEBUG_context.Esp;
115         case REG_CS:  return DEBUG_context.SegCs;
116         case REG_DS:  return DEBUG_context.SegDs;
117         case REG_ES:  return DEBUG_context.SegEs;
118         case REG_SS:  return DEBUG_context.SegSs;
119         case REG_FS:  return DEBUG_context.SegFs;
120         case REG_GS:  return DEBUG_context.SegGs;
121         case REG_AX:  return LOWORD(DEBUG_context.Eax);
122         case REG_BX:  return LOWORD(DEBUG_context.Ebx);
123         case REG_CX:  return LOWORD(DEBUG_context.Ecx);
124         case REG_DX:  return LOWORD(DEBUG_context.Edx);
125         case REG_SI:  return LOWORD(DEBUG_context.Esi);
126         case REG_DI:  return LOWORD(DEBUG_context.Edi);
127         case REG_BP:  return LOWORD(DEBUG_context.Ebp);
128         case REG_FL:  return LOWORD(DEBUG_context.EFlags);
129         case REG_IP:  return LOWORD(DEBUG_context.Eip);
130         case REG_SP:  return LOWORD(DEBUG_context.Esp);
131     }
132 #endif
133     return 0;  /* should not happen */
134 }
135
136
137 /***********************************************************************
138  *           DEBUG_Flags
139  *
140  * Return Flag String.
141  */
142 char *DEBUG_Flags( DWORD flag, char *buf )
143 {
144     char *pt;
145
146     strcpy( buf, "   - 00      - - - " );
147     pt = buf + strlen( buf );
148     if ( buf >= pt-- ) return( buf );
149     if ( flag & 0x00000001 ) *pt = 'C'; /* Carry Falg */
150     if ( buf >= pt-- ) return( buf );
151     if ( flag & 0x00000002 ) *pt = '1';
152     if ( buf >= pt-- ) return( buf );
153     if ( flag & 0x00000004 ) *pt = 'P'; /* Parity Flag */
154     if ( buf >= pt-- ) return( buf );
155     if ( flag & 0x00000008 ) *pt = '-';
156     if ( buf >= pt-- ) return( buf );
157     if ( flag & 0x00000010 ) *pt = 'A'; /* Auxiliary Carry Flag */
158     if ( buf >= pt-- ) return( buf );
159     if ( flag & 0x00000020 ) *pt = '-';
160     if ( buf >= pt-- ) return( buf );
161     if ( flag & 0x00000040 ) *pt = 'Z'; /* Zero Flag */
162     if ( buf >= pt-- ) return( buf );
163     if ( flag & 0x00000080 ) *pt = 'S'; /* Sign Flag */
164     if ( buf >= pt-- ) return( buf );
165     if ( flag & 0x00000100 ) *pt = 'T'; /* Trap/Trace Flag */
166     if ( buf >= pt-- ) return( buf );
167     if ( flag & 0x00000200 ) *pt = 'I'; /* Interupt Enable Flag */
168     if ( buf >= pt-- ) return( buf );
169     if ( flag & 0x00000400 ) *pt = 'D'; /* Direction Indicator */
170     if ( buf >= pt-- ) return( buf );
171     if ( flag & 0x00000800 ) *pt = 'O'; /* Overflow Flag */
172     if ( buf >= pt-- ) return( buf );
173     if ( flag & 0x00001000 ) *pt = '1'; /* I/O Privilage Level */
174     if ( buf >= pt-- ) return( buf );
175     if ( flag & 0x00002000 ) *pt = '1'; /* I/O Privilage Level */
176     if ( buf >= pt-- ) return( buf );
177     if ( flag & 0x00004000 ) *pt = 'N'; /* Nested Task Flag */
178     if ( buf >= pt-- ) return( buf );
179     if ( flag & 0x00008000 ) *pt = '-';
180     if ( buf >= pt-- ) return( buf );
181     if ( flag & 0x00010000 ) *pt = 'R'; /* Resume Flag */
182     if ( buf >= pt-- ) return( buf );
183     if ( flag & 0x00020000 ) *pt = 'V'; /* Vritual Mode Flag */
184     if ( buf >= pt-- ) return( buf );
185     if ( flag & 0x00040000 ) *pt = 'a'; /* Alignment Check Flag */
186     if ( buf >= pt-- ) return( buf );
187     return( buf );
188 }
189
190
191 /***********************************************************************
192  *           DEBUG_InfoRegisters
193  *
194  * Display registers information. 
195  */
196 void DEBUG_InfoRegisters(void)
197 {
198     fprintf(stderr,"Register dump:\n");
199
200 #ifdef __i386__
201     /* First get the segment registers out of the way */
202     fprintf( stderr," CS:%04x SS:%04x DS:%04x ES:%04x FS:%04x GS:%04x",
203              (WORD)DEBUG_context.SegCs, (WORD)DEBUG_context.SegSs,
204              (WORD)DEBUG_context.SegDs, (WORD)DEBUG_context.SegEs,
205              (WORD)DEBUG_context.SegFs, (WORD)DEBUG_context.SegGs );
206     if (DEBUG_CurrThread->dbg_mode == 16)
207     {
208         char flag[33];
209
210         fprintf( stderr,"\n IP:%04x SP:%04x BP:%04x FLAGS:%04x(%s)\n",
211                  LOWORD(DEBUG_context.Eip), LOWORD(DEBUG_context.Esp),
212                  LOWORD(DEBUG_context.Ebp), LOWORD(DEBUG_context.EFlags),
213                  DEBUG_Flags(LOWORD(DEBUG_context.EFlags), flag));
214         fprintf( stderr," AX:%04x BX:%04x CX:%04x DX:%04x SI:%04x DI:%04x\n",
215                  LOWORD(DEBUG_context.Eax), LOWORD(DEBUG_context.Ebx),
216                  LOWORD(DEBUG_context.Ecx), LOWORD(DEBUG_context.Edx),
217                  LOWORD(DEBUG_context.Esi), LOWORD(DEBUG_context.Edi) );
218     }
219     else  /* 32-bit mode */
220     {
221         char flag[33];
222
223         fprintf( stderr, "\n EIP:%08lx ESP:%08lx EBP:%08lx EFLAGS:%08lx(%s)\n", 
224                  DEBUG_context.Eip, DEBUG_context.Esp,
225                  DEBUG_context.Ebp, DEBUG_context.EFlags,
226                  DEBUG_Flags(DEBUG_context.EFlags, flag));
227         fprintf( stderr, " EAX:%08lx EBX:%08lx ECX:%08lx EDX:%08lx\n", 
228                  DEBUG_context.Eax, DEBUG_context.Ebx,
229                  DEBUG_context.Ecx, DEBUG_context.Edx );
230         fprintf( stderr, " ESI:%08lx EDI:%08lx\n",
231                  DEBUG_context.Esi, DEBUG_context.Edi );
232     }
233 #endif
234 }
235
236
237 /***********************************************************************
238  *           DEBUG_ValidateRegisters
239  *
240  * Make sure all registers have a correct value for returning from
241  * the signal handler.
242  */
243 BOOL DEBUG_ValidateRegisters(void)
244 {
245 #ifdef __i386__
246     if (DEBUG_context.EFlags & V86_FLAG) return TRUE;
247
248 #if 0
249 /* Check that a selector is a valid ring-3 LDT selector, or a NULL selector */
250 #define CHECK_SEG(seg,name) \
251     if (((seg) & ~3) && ((((seg) & 7) != 7) || !DEBUG_IsSelector(seg))) { \
252         fprintf( stderr, "*** Invalid value for %s register: %04x\n", \
253                  (name), (WORD)(seg) ); \
254         return FALSE; \
255     }
256
257     cs = __get_cs();
258     ds = __get_ds();
259     if (CS_reg(DEBUG_context) != cs) CHECK_SEG(CS_reg(DEBUG_context), "CS");
260     if (SS_reg(DEBUG_context) != ds) CHECK_SEG(SS_reg(DEBUG_context), "SS");
261     if (DS_reg(DEBUG_context) != ds) CHECK_SEG(DS_reg(DEBUG_context), "DS");
262     if (ES_reg(DEBUG_context) != ds) CHECK_SEG(ES_reg(DEBUG_context), "ES");
263     if (FS_reg(DEBUG_context) != ds) CHECK_SEG(FS_reg(DEBUG_context), "FS");
264     if (GS_reg(DEBUG_context) != ds) CHECK_SEG(GS_reg(DEBUG_context), "GS");
265 #endif
266
267     /* Check that CS and SS are not NULL */
268
269     if (!(DEBUG_context.SegCs & ~3))
270     {
271         fprintf( stderr, "*** Invalid value for CS register: %04x\n",
272                  (WORD)DEBUG_context.SegCs );
273         return FALSE;
274     }
275     if (!(DEBUG_context.SegSs & ~3))
276     {
277         fprintf( stderr, "*** Invalid value for SS register: %04x\n",
278                  (WORD)DEBUG_context.SegSs );
279         return FALSE;
280     }
281     return TRUE;
282 #undef CHECK_SEG
283 #else
284     return TRUE;
285 #endif
286 }