6e7a0003d86ff5f5971010792233527a402ca9f0
[tpg/acess2.git] / Kernel / debug.c
1 /*
2  * AcessOS Microkernel Version
3  * debug.c
4  * 
5  * TODO: Move the Debug_putchar methods out to the arch/ tree
6  */
7 #include <acess.h>
8 #include <stdarg.h>
9
10 #define DEBUG_MAX_LINE_LEN      256
11
12 #define LOCK_DEBUG_OUTPUT       1
13
14 // === IMPORTS ===
15 extern void Threads_Dump(void);
16 extern void     KernelPanic_SetMode(void);
17 extern void     KernelPanic_PutChar(char Ch);
18
19 // === PROTOTYPES ===
20 static void     Debug_Putchar(char ch);
21 static void     Debug_Puts(int DbgOnly, const char *Str);
22 void    Debug_DbgOnlyFmt(const char *format, va_list args);
23 void    Debug_FmtS(const char *format, ...);
24 void    Debug_Fmt(const char *format, va_list args);
25 void    Debug_SetKTerminal(const char *File);
26
27 // === GLOBALS ===
28  int    gDebug_Level = 0;
29  int    giDebug_KTerm = -1;
30  int    gbDebug_IsKPanic = 0;
31 volatile int    gbInPutChar = 0;
32 #if LOCK_DEBUG_OUTPUT
33 tShortSpinlock  glDebug_Lock;
34 #endif
35
36 // === CODE ===
37 static void Debug_Putchar(char ch)
38 {       
39         Debug_PutCharDebug(ch);
40         if( !gbDebug_IsKPanic )
41         {
42                 if(gbInPutChar) return ;
43                 gbInPutChar = 1;
44                 if(giDebug_KTerm != -1)
45                         VFS_Write(giDebug_KTerm, 1, &ch);
46                 gbInPutChar = 0;
47         }
48         else
49                 KernelPanic_PutChar(ch);
50 }
51
52 static void Debug_Puts(int UseKTerm, const char *Str)
53 {
54          int    len = 0;
55         
56         Debug_PutStringDebug(Str);
57         
58         if( gbDebug_IsKPanic )
59         {               
60                 for( len = 0; Str[len]; len ++ )
61                         KernelPanic_PutChar( Str[len] );
62         }
63         else
64                 for( len = 0; Str[len]; len ++ );
65         
66         // Output to the kernel terminal
67         if( UseKTerm && !gbDebug_IsKPanic && giDebug_KTerm != -1)
68         {
69                 if(gbInPutChar) return ;
70                 gbInPutChar = 1;
71                 VFS_Write(giDebug_KTerm, len, Str);
72                 gbInPutChar = 0;
73         }
74 }
75
76 void Debug_DbgOnlyFmt(const char *format, va_list args)
77 {
78         char    buf[DEBUG_MAX_LINE_LEN];
79          int    len;
80         buf[DEBUG_MAX_LINE_LEN-1] = 0;
81         len = vsnprintf(buf, DEBUG_MAX_LINE_LEN-1, format, args);
82         //if( len < DEBUG_MAX_LINE )
83                 // do something
84         Debug_Puts(0, buf);
85 }
86
87 void Debug_Fmt(const char *format, va_list args)
88 {
89         char    buf[DEBUG_MAX_LINE_LEN];
90          int    len;
91         buf[DEBUG_MAX_LINE_LEN-1] = 0;
92         len = vsnprintf(buf, DEBUG_MAX_LINE_LEN-1, format, args);
93         //if( len < DEBUG_MAX_LINE )
94                 // do something
95         Debug_Puts(1, buf);
96         return ;
97 }
98
99 void Debug_FmtS(const char *format, ...)
100 {
101         va_list args;   
102         va_start(args, format);
103         Debug_Fmt(format, args);
104         va_end(args);
105 }
106
107 void Debug_KernelPanic()
108 {
109         gbDebug_IsKPanic = 1;
110         KernelPanic_SetMode();
111 }
112
113 /**
114  * \fn void LogF(const char *Msg, ...)
115  * \brief Raw debug log (no new line, no prefix)
116  */
117 void LogF(const char *Fmt, ...)
118 {
119         va_list args;
120
121         #if LOCK_DEBUG_OUTPUT
122         SHORTLOCK(&glDebug_Lock);
123         #endif
124         
125         va_start(args, Fmt);
126
127         Debug_Fmt(Fmt, args);
128
129         va_end(args);
130         
131         #if LOCK_DEBUG_OUTPUT
132         SHORTREL(&glDebug_Lock);
133         #endif
134 }
135 /**
136  * \fn void Debug(const char *Msg, ...)
137  * \brief Print only to the debug channel (not KTerm)
138  */
139 void Debug(const char *Fmt, ...)
140 {
141         va_list args;
142         
143         #if LOCK_DEBUG_OUTPUT
144         SHORTLOCK(&glDebug_Lock);
145         #endif
146
147         Debug_Puts(0, "Debug: ");
148         va_start(args, Fmt);
149         Debug_DbgOnlyFmt(Fmt, args);
150         va_end(args);
151         Debug_PutCharDebug('\r');
152         Debug_PutCharDebug('\n');
153         #if LOCK_DEBUG_OUTPUT
154         SHORTREL(&glDebug_Lock);
155         #endif
156 }
157 /**
158  * \fn void Log(const char *Msg, ...)
159  */
160 void Log(const char *Fmt, ...)
161 {
162         va_list args;
163         
164         #if LOCK_DEBUG_OUTPUT
165         SHORTLOCK(&glDebug_Lock);
166         #endif
167
168         Debug_Puts(1, "Log: ");
169         va_start(args, Fmt);
170         Debug_Fmt(Fmt, args);
171         va_end(args);
172         Debug_Putchar('\r');
173         Debug_Putchar('\n');
174         
175         #if LOCK_DEBUG_OUTPUT
176         SHORTREL(&glDebug_Lock);
177         #endif
178 }
179 void Warning(const char *Fmt, ...)
180 {
181         va_list args;
182         
183         #if LOCK_DEBUG_OUTPUT
184         SHORTLOCK(&glDebug_Lock);
185         #endif
186         
187         Debug_Puts(1, "Warning: ");
188         va_start(args, Fmt);
189         Debug_Fmt(Fmt, args);
190         va_end(args);
191         Debug_Putchar('\r');
192         Debug_Putchar('\n');
193         
194         #if LOCK_DEBUG_OUTPUT
195         SHORTREL(&glDebug_Lock);
196         #endif
197 }
198 void Panic(const char *Fmt, ...)
199 {
200         va_list args;
201         
202         #if LOCK_DEBUG_OUTPUT
203         SHORTLOCK(&glDebug_Lock);
204         #endif
205         // And never SHORTREL
206         
207         Debug_KernelPanic();
208         
209         Debug_Puts(1, "Panic: ");
210         va_start(args, Fmt);
211         Debug_Fmt(Fmt, args);
212         va_end(args);
213         Debug_Putchar('\r');
214         Debug_Putchar('\n');
215
216         Threads_Dump();
217
218 //      __asm__ __volatile__ ("xchg %bx, %bx");
219 //      __asm__ __volatile__ ("cli;\n\thlt");
220 //      for(;;) __asm__ __volatile__ ("hlt");
221         for(;;) ;
222 }
223
224 void Debug_SetKTerminal(const char *File)
225 {
226          int    tmp;
227         if(giDebug_KTerm != -1) {
228                 tmp = giDebug_KTerm;
229                 giDebug_KTerm = -1;
230                 VFS_Close(tmp);
231         }
232         tmp = VFS_Open(File, VFS_OPENFLAG_WRITE);
233         Log_Log("Debug", "Opened '%s' as 0x%x", File, tmp);
234         giDebug_KTerm = tmp;
235         Log_Log("Debug", "Returning to %p", __builtin_return_address(0));
236 }
237
238 void Debug_Enter(const char *FuncName, const char *ArgTypes, ...)
239 {
240         va_list args;
241          int    i;
242          int    pos;
243         tTID    tid = Threads_GetTID();
244          
245         #if LOCK_DEBUG_OUTPUT
246         SHORTLOCK(&glDebug_Lock);
247         #endif
248
249         i = gDebug_Level ++;
250
251         va_start(args, ArgTypes);
252
253         LogF("%012lli ", now());
254         while(i--)      Debug_Putchar(' ');
255
256         Debug_Puts(1, FuncName);
257         Debug_FmtS("[%i]", tid);
258         Debug_Puts(1, ": (");
259
260         while(*ArgTypes)
261         {
262                 pos = strpos(ArgTypes, ' ');
263                 if(pos == -1 || pos > 1) {
264                         if(pos == -1)
265                                 Debug_Puts(1, ArgTypes+1);
266                         else {
267                                 for( i = 1; i < pos; i ++ )
268                                         Debug_Putchar(ArgTypes[i]);
269                         }
270                         Debug_Putchar('=');
271                 }
272                 switch(*ArgTypes)
273                 {
274                 case 'p':       LogF("%p", va_arg(args, void*));        break;
275                 case 's':       LogF("'%s'", va_arg(args, char*));      break;
276                 case 'i':       LogF("%i", va_arg(args, int));  break;
277                 case 'u':       LogF("%u", va_arg(args, Uint)); break;
278                 case 'x':       LogF("0x%x", va_arg(args, Uint));       break;
279                 case 'b':       LogF("0b%b", va_arg(args, Uint));       break;
280                 case 'X':       LogF("0x%llx", va_arg(args, Uint64));   break;  // Extended (64-Bit)
281                 case 'B':       LogF("0b%llb", va_arg(args, Uint64));   break;  // Extended (64-Bit)
282                 }
283                 if(pos != -1) {
284                         Debug_Putchar(',');     Debug_Putchar(' ');
285                 }
286
287                 if(pos == -1)   break;
288                 ArgTypes = &ArgTypes[pos+1];
289         }
290
291         va_end(args);
292         Debug_Putchar(')');     Debug_Putchar('\r');    Debug_Putchar('\n');
293         
294         #if LOCK_DEBUG_OUTPUT
295         SHORTREL(&glDebug_Lock);
296         #endif
297 }
298
299 void Debug_Log(const char *FuncName, const char *Fmt, ...)
300 {
301         va_list args;
302          int    i = gDebug_Level;
303         tTID    tid = Threads_GetTID();
304
305         #if LOCK_DEBUG_OUTPUT
306         SHORTLOCK(&glDebug_Lock);
307         #endif
308
309         va_start(args, Fmt);
310
311         LogF("%012lli ", now());
312         while(i--)      Debug_Putchar(' ');
313
314         Debug_Puts(1, FuncName);
315         Debug_FmtS("[%i]", tid);
316         Debug_Puts(1, ": ");
317         Debug_Fmt(Fmt, args);
318
319         va_end(args);
320         Debug_Putchar('\r');
321         Debug_Putchar('\n');
322         
323         #if LOCK_DEBUG_OUTPUT
324         SHORTREL(&glDebug_Lock);
325         #endif
326 }
327
328 void Debug_Leave(const char *FuncName, char RetType, ...)
329 {
330         va_list args;
331          int    i;
332         tTID    tid = Threads_GetTID();
333
334         #if LOCK_DEBUG_OUTPUT
335         SHORTLOCK(&glDebug_Lock);
336         #endif
337         
338         i = --gDebug_Level;
339
340         va_start(args, RetType);
341
342         if( i == -1 ) {
343                 gDebug_Level = 0;
344                 i = 0;
345         }
346         LogF("%012lli ", now());
347         // Indenting
348         while(i--)      Debug_Putchar(' ');
349
350         Debug_Puts(1, FuncName);
351         Debug_FmtS("[%i]", tid);
352         Debug_Puts(1, ": RETURN");
353
354         // No Return
355         if(RetType == '-') {
356                 Debug_Putchar('\r');
357                 Debug_Putchar('\n');
358                 #if LOCK_DEBUG_OUTPUT
359                 SHORTREL(&glDebug_Lock);
360                 #endif
361                 return;
362         }
363
364         Debug_Putchar(' ');
365         switch(RetType)
366         {
367         case 'n':       Debug_Puts(1, "NULL");  break;
368         case 'p':       Debug_Fmt("%p", args);  break;
369         case 's':       Debug_Fmt("'%s'", args);        break;
370         case 'i':       Debug_Fmt("%i", args);  break;
371         case 'u':       Debug_Fmt("%u", args);  break;
372         case 'x':       Debug_Fmt("0x%x", args);        break;
373         // Extended (64-Bit)
374         case 'X':       Debug_Fmt("0x%llx", args);      break;
375         }
376         Debug_Putchar('\r');
377         Debug_Putchar('\n');
378
379         va_end(args);
380         
381         #if LOCK_DEBUG_OUTPUT
382         SHORTREL(&glDebug_Lock);
383         #endif
384 }
385
386 void Debug_HexDump(const char *Header, const void *Data, Uint Length)
387 {
388         const Uint8     *cdat = Data;
389         Uint    pos = 0;
390         Debug_Puts(1, Header);
391         LogF(" (Hexdump of %p)\r\n", Data);
392
393         #define CH(n)   ((' '<=cdat[(n)]&&cdat[(n)]<=0x7F) ? cdat[(n)] : '.')
394
395         while(Length >= 16)
396         {
397                 Log("%04x: %02x %02x %02x %02x %02x %02x %02x %02x"
398                         " %02x %02x %02x %02x %02x %02x %02x %02x"
399                         "  %c%c%c%c%c%c%c%c %c%c%c%c%c%c%c%c",
400                         pos,
401                         cdat[0], cdat[1], cdat[2], cdat[3], cdat[4], cdat[5], cdat[6], cdat[7],
402                         cdat[8], cdat[9], cdat[10], cdat[11], cdat[12], cdat[13], cdat[14], cdat[15],
403                         CH(0),  CH(1),  CH(2),  CH(3),  CH(4),  CH(5),  CH(6),  CH(7),
404                         CH(8),  CH(9),  CH(10), CH(11), CH(12), CH(13), CH(14), CH(15)
405                         );
406                 Length -= 16;
407                 cdat += 16;
408                 pos += 16;
409         }
410
411         {
412                  int    i ;
413                 LogF("Log: %04x: ", pos);
414                 for(i = 0; i < Length; i ++)
415                 {
416                         LogF("%02x ", cdat[i]);
417                 }
418                 for( ; i < 16; i ++)    LogF("   ");
419                 LogF(" ");
420                 for(i = 0; i < Length; i ++)
421                 {
422                         if( i == 8 )    LogF(" ");
423                         LogF("%c", CH(i));
424                 }
425         }
426         
427         Debug_Putchar('\r');
428         Debug_Putchar('\n');
429 }
430
431 // --- EXPORTS ---
432 EXPORT(Debug);
433 EXPORT(Log);
434 EXPORT(Warning);
435 EXPORT(Debug_Enter);
436 EXPORT(Debug_Log);
437 EXPORT(Debug_Leave);

UCC git Repository :: git.ucc.asn.au