diff options
Diffstat (limited to 'modules/mod_debug/mod_debug.c')
-rw-r--r-- | modules/mod_debug/mod_debug.c | 2882 |
1 files changed, 2882 insertions, 0 deletions
diff --git a/modules/mod_debug/mod_debug.c b/modules/mod_debug/mod_debug.c new file mode 100644 index 0000000..4ea03d6 --- /dev/null +++ b/modules/mod_debug/mod_debug.c @@ -0,0 +1,2882 @@ +/* + * Copyright © 2006-2016 SplinterGU (Fenix/Bennugd) + * Copyright © 2002-2006 Fenix Team (Fenix) + * Copyright © 1999-2002 José Luis Cebrián Pagüe (Fenix) + * + * This file is part of Bennu - Game Development + * + * This software is provided 'as-is', without any express or implied + * warranty. In no event will the authors be held liable for any damages + * arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, + * including commercial applications, and to alter it and redistribute it + * freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not + * claim that you wrote the original software. If you use this software + * in a product, an acknowledgment in the product documentation would be + * appreciated but is not required. + * + * 2. Altered source versions must be plainly marked as such, and must not be + * misrepresented as being the original software. + * + * 3. This notice may not be removed or altered from any source + * distribution. + * + */ + +/* --------------------------------------------------------------------------- */ + +#include <stdio.h> +#include <stdarg.h> +#include <stdlib.h> +#include <string.h> +#include <ctype.h> + +#include <SDL.h> + +#include "bgdcore.h" +#include "bgdrtm.h" +#include "bgddl.h" +#include "dlvaracc.h" + +#include "dcb.h" +#include "instance.h" + +#include "files.h" + +#include "xstrings.h" +#include "libkey.h" +#include "libgrbase.h" +#include "librender.h" + +/* --------------------------------------------------------------------------- */ + +extern void systext_color( int cfg, int cbg ); +extern void systext_puts( GRAPH * map, int x, int y, char * str, int len ); + +/* --------------------------------------------------------------------------- */ + +#ifndef _vsnprintf +#define _vsnprintf vsnprintf +#endif + +#ifndef _snprintf +#define _snprintf snprintf +#endif + +/* --------------------------------------------------------------------------- */ + +enum { + PROCESS_ID = 0, + FATHER, + BIGBRO, + SON, + STATUS +}; + +enum { + SHIFTSTATUS = 0 +}; + +/* --------------------------------------------------------------------------- */ + +DLVARFIXUP __bgdexport( mod_debug, locals_fixup )[] = +{ + { "id" , NULL, -1, -1 }, + { "father" , NULL, -1, -1 }, + { "bigbro" , NULL, -1, -1 }, + { "son" , NULL, -1, -1 }, + { "reserved.status" , NULL, -1, -1 }, + + { NULL , NULL, -1, -1 } +}; + +/* --------------------------------------------------------------------------- */ + +DLVARFIXUP __bgdexport( mod_debug, globals_fixup )[] = +{ + { "shift_status" , NULL, -1, -1 }, + + { NULL , NULL, -1, -1 } +}; + +/* --------------------------------------------------------------------------- */ + +#define MAXTEXT 5000 + +#define CHARWIDTH 6 +#define CHARHEIGHT 8 + +#define CONSOLE_HISTORY 512 +#define CONSOLE_LINES 25 +#define CONSOLE_COLUMNS 80 +#define COMMAND_HISTORY 128 + +#define MAX_EXPRESSIONS 128 + +/* --------------------------------------------------------------------------- */ + +#define HELPTXT \ + "¬04Process Info¬07\n" \ + "¬05INSTANCES ¬07 List all running processes\n" \ + "¬05GLOBALS ¬07 Show global vars with values\n" \ + "¬05LOCALS proc ¬07 Show a process's local vars\n" \ + "¬05PRIVATES proc ¬07 Show a process's private vars\n" \ + "¬05PUBLICS proc ¬07 Show a process's public vars\n" \ + "\n" \ + "¬04Execution Commands¬07\n" \ + "¬05GO ¬07 Continue the execution\n" \ + "¬05TRACE ¬07 Execute one instruction and Debug\n" \ + "¬05NEXTFRAME ¬07 Continue to next frame\n" \ + "¬05NEXTPROC ¬07 Continue to next process\n" \ + "\n" \ + "¬04Breakpoints¬07\n" \ + "¬05BREAK ¬07 List breakpoints\n" \ + "¬05BREAK proc ¬07 Set a breakpoint on process\n" \ + "¬05BREAKALL ¬07 Set breakpoints on all processes\n" \ + "¬05BREAKALLTYPES ¬07 Set breakpoints on all processes types\n" \ + "¬05DELETE proc ¬07 Delete a breakpoint\n" \ + "¬05DELETEALL ¬07 Delete all breakpoints on all processess\n" \ + "¬05DELETEALLTYPES ¬07 Delete all breakpoints on all processess types\n" \ + "\n" \ + "¬04Process Interaction¬07\n" \ + "¬05RUN proc [args] ¬07 Run a process\n" \ + "¬05KILL proc ¬07 Kill a process\n" \ + "¬05WAKEUP proc ¬07 Wakeup a process\n" \ + "¬05SLEEP proc ¬07 Sleep a process\n" \ + "¬05FREEZE proc ¬07 Freeze a process\n" \ + "¬05KILLALL proc ¬07 Kill all process with criteria\n" \ + "¬05WAKEUPALL proc ¬07 Wakeup all process with criteria\n" \ + "¬05SLEEPALL proc ¬07 Sleep all process with criteria\n" \ + "¬05FREEZEALL proc ¬07 Freeze all process with criteria\n" \ + "\n" \ + "¬04Misc¬07\n" \ + "¬05SHOW expression ¬07 Evaluate and show some expression\n" \ + "¬05SHOW ¬07 List all expressions to show\n" \ + "¬05SHOWDEL ExpID ¬07 Delete an expression (by ID, use SHOW)\n" \ + "¬05SHOWDELALL ¬07 Delete all expression\n" \ + "¬05STRINGS ¬07 Show all strings in memory\n" \ + "¬05VARS ¬07 Show internals vars\n" \ + "¬05QUIT ¬07 Kill the program and exit\n" \ + "\n" \ + "¬04Keys¬07\n" \ + "¬05ESC ¬07 Cancel command\n" \ + "¬05UP/DOWN ¬07 Command history navigation\n" \ + "¬05PGUP/PGDN ¬07 Page Up/Page Down\n" \ + "¬05CTRL+CURSORS ¬07 Console scroll\n" \ + "¬05ALT+CURSORS ¬07 Console window size\n" \ + "¬05SHIFT+CURSORS ¬07 List window scroll\n" \ + "\n" \ + "You can evaluate free expressions in the console, and you can see/change\n" \ + "local, public and private vars using the '.' operator\n" \ + "(pe: 65535.X, MAIN.X, etc.)\n\n" + +/* --------------------------------------------------------------------------- */ + +#define HOTKEYHELP_SIZE 50 + +#define HOTKEYHELP1 "¬01F1:¬00?\x03¬01F2:¬00Procs\x03¬01F5:¬00Go\x03¬01F8:¬00Trace\x03¬01F10:¬00NFrame\x03¬01F11:¬00NProc" +#define HOTKEYHELP2 "¬01F1:¬00?\x03¬01F2:¬00Brief\x03¬01F6:¬00Procs\x03¬01F9:¬00Break" +#define HOTKEYHELP3 "¬01F1:¬00?\x03¬01F2:¬00Brief\x03¬01F3:¬00Loc\x03¬01F4:¬00Pri\x03¬01F5:¬00Pub\x03¬01F6:¬00Types\x03¬01F9:¬00Brk" + +/* --------------------------------------------------------------------------- */ + +#define N_CONSOLE_VARS (sizeof(console_vars)/sizeof(console_vars[0])) + +#define CON_DWORD 0x0001 +#define CON_DWORD_HEX 0x8001 + +/* --------------------------------------------------------------------------- */ + +/* Tipos de token */ +#define IDENTIFIER 1 +#define STRING 2 +#define NUMBER 3 +#define OPERATOR 4 +#define NOTOKEN 5 + +/* --------------------------------------------------------------------------- */ + +static struct +{ + int type ; + DCB_VAR var ; + double value ; + void * data ; + char name[256] ; +} result, lvalue ; + +static struct +{ + enum { T_ERROR, T_VARIABLE, T_NUMBER, T_CONSTANT, T_STRING } type ; + char name[128] ; + double code ; +} token ; + +static const char * token_ptr ; + +/* --------------------------------------------------------------------------- */ + +/* Console contents */ +static int console_lines = CONSOLE_LINES ; +static int console_columns = CONSOLE_COLUMNS ; + +static char * console[CONSOLE_HISTORY] ; +static int console_initialized = 0 ; +static int console_head = 0 ; +static int console_tail = 0 ; + +static char * command[COMMAND_HISTORY] ; +static int command_initialized = 0 ; +static int command_head = 0 ; +static int command_tail = 0 ; +static int command_count = 0 ; + +static char * show_expression[MAX_EXPRESSIONS] = { NULL }; +static int show_expression_count = 0; +static int console_showcolor = 0xffffff; + +static int console_y = 0 ; + +/* --------------------------------------------------------------------------- */ + +static int debug_on_frame = 0; +static int break_on_next_proc = 0; + +/* --------------------------------------------------------------------------- */ + +static struct +{ + char * name ; + void * value ; + int type ; +} +console_vars[] = +{ + { "SHOW_COLOR", &console_showcolor, CON_DWORD_HEX }, + { "FILES", &opened_files, CON_DWORD }, + { "DEBUG_LEVEL", &debug, CON_DWORD }, +} ; + +/* --------------------------------------------------------------------------- */ + +static int console_showing = 0 ; +static int console_scroll_pos = 0 ; +static int console_scroll_lateral_pos = 0 ; +static char console_input[128] ; + +/* --------------------------------------------------------------------------- */ + +static void eval_immediate(); +static void eval_value(); +static void eval_factor(); +static void eval_subexpression(); +static char * eval_expression( const char * here, int interactive ); + +static char * describe_type( DCB_TYPEDEF type, int from ); +static char * show_value( DCB_TYPEDEF type, void * data ); +static void show_struct( int num, char * title, int indent, void * data ); +static void show_var( DCB_VAR var, char * name, void * data, char * title, int indent ); + +static int type_size( DCB_TYPEDEF orig ); +static DCB_TYPEDEF reduce_type( DCB_TYPEDEF orig ); +static void var2const(); + +static void console_do( const char * command ); + +/* --------------------------------------------------------------------------- */ + +static void console_scroll( int direction ) +{ + console_scroll_pos += direction ; + if ( direction < 0 ) + { + if ( console_scroll_pos < 0 ) console_scroll_pos = 0 ; + } + else + { + if ( console_scroll_pos > CONSOLE_HISTORY ) console_scroll_pos = CONSOLE_HISTORY ; + } +} + +/* --------------------------------------------------------------------------- */ + +static void console_putline( char * text ) +{ + if ( !console_initialized ) + { + memset( console, 0, sizeof( console ) ) ; + console_initialized = 1 ; + } + + if ( console[console_tail] ) free( console[console_tail] ) ; + console[console_tail] = strdup( text ) ; + + console_tail++ ; + if ( console_tail == CONSOLE_HISTORY ) console_tail = 0 ; + if ( console_tail == console_head ) + { + console_head++ ; + if ( console_head == CONSOLE_HISTORY ) console_head = 0 ; + } +} + +/* --------------------------------------------------------------------------- */ + +static void console_printf( const char *fmt, ... ) +{ + char text[MAXTEXT], * ptr, * iptr ; + + va_list ap; + va_start( ap, fmt ); + _vsnprintf( text, sizeof( text ), fmt, ap ); + va_end( ap ); + text[sizeof( text )-1] = 0; + + if ( *text == '[' ) + { + memmove( text + 3, text, strlen( text ) + 1 ) ; + memmove( text, "¬08", 3 ) ; + ptr = strchr( text, ']' ) ; + if ( ptr ) + { + ptr++ ; + memmove( ptr + 3, ptr, strlen( ptr ) + 1 ) ; + memmove( ptr, "¬07", 3 ) ; + } + } + + iptr = text ; + ptr = text ; + + while ( *ptr ) + { + if ( *ptr == '\n' ) + { + *ptr = 0 ; + console_putline( iptr ) ; + iptr = ptr + 1 ; + } + if ( *ptr == '¬' ) + { + ptr++ ; + if ( isdigit( *ptr ) ) ptr++ ; + if ( isdigit( *ptr ) ) ptr++ ; + continue ; + } + ptr++ ; + } + + if ( ptr > iptr ) console_putline( iptr ) ; +} + +/* --------------------------------------------------------------------------- */ + +static void console_putcommand( const char * commandline ) +{ + if ( !command_initialized ) + { + memset( command, 0, sizeof( command ) ); + command_initialized = 1; + } + + if ( command[command_tail] ) free( command[command_tail] ); + command[command_tail++] = strdup( commandline ); + if ( command_tail == COMMAND_HISTORY ) command_tail = 0; + if ( command_tail == command_head ) + { + if ( ++command_head == COMMAND_HISTORY ) command_head = 0; + } + else + command_count++; +} + +/* --------------------------------------------------------------------------- */ + +static const char * console_getcommand( int offset ) +{ + if ( offset >= 0 || offset < -command_count ) return NULL; + + offset = command_tail + offset; + while ( offset < 0 ) offset = COMMAND_HISTORY + offset; + return command[offset]; +} + +/* --------------------------------------------------------------------------- */ + +static void console_getkey( int key, int sym ) +{ + static int history_offset = 0; + char buffer[2] ; + const char * command; + + if ( !key ) + { + if ( sym == SDLK_UP ) + { + command = console_getcommand( --history_offset ); + if ( command == NULL ) + history_offset++; + else + strncpy( console_input, command, 127 ); + } + + if ( sym == SDLK_DOWN ) + { + if ( history_offset == -1 ) + { + *console_input = 0; + history_offset++; + } + else + { + command = console_getcommand( ++history_offset ); + if ( command == NULL ) + history_offset--; + else + strncpy( console_input, command, 127 ); + } + } + } + + if ( key == SDLK_BACKSPACE && *console_input ) console_input[strlen( console_input )-1] = 0 ; + if ( key == SDLK_RCTRL ) *console_input = 0 ; + if ( key == SDLK_RETURN ) + { + + console_scroll_pos = 0 ; + console_printf( "¬07> %s", console_input ) ; + if ( * console_input ) + { + console_putcommand( console_input ); + console_do( console_input ) ; + *console_input = 0 ; + history_offset = 0; + } + } + + if ( key >= SDLK_SPACE && key <= SDLK_WORLD_95 ) + { + buffer[0] = key ; + buffer[1] = 0 ; + strcat( console_input, buffer ) ; + } +} + +/* --------------------------------------------------------------------------- */ + +static void console_lateral_scroll( int direction ) +{ + if ( direction > 0 ) + { + console_scroll_lateral_pos-- ; + if ( console_scroll_lateral_pos < 0 ) console_scroll_lateral_pos = 0 ; + } + else + { + console_scroll_lateral_pos++ ; + if ( console_scroll_lateral_pos > MAXTEXT ) console_scroll_lateral_pos = MAXTEXT ; + } +} + +/* --------------------------------------------------------------------------- */ + +static char * describe_type( DCB_TYPEDEF type, int from ) +{ + static char buffer[512] ; + int i ; + + if ( !from ) buffer[0] = 0 ; + + switch ( type.BaseType[from] ) + { + case TYPE_ARRAY: + for ( i = from ; type.BaseType[i] == TYPE_ARRAY; i++ ) ; + describe_type( type, i ) ; + for ( i = from ; type.BaseType[i] == TYPE_ARRAY; i++ ) + _snprintf( buffer + strlen( buffer ), 512 - strlen( buffer ), "[%d]", type.Count[i] - 1 ) ; + break ; + + case TYPE_STRUCT: + strcat( buffer, "STRUCT" ) ; + break ; + + case TYPE_DWORD: + strcat( buffer, "DWORD" ) ; + break ; + + case TYPE_INT: + strcat( buffer, "INT" ) ; + break ; + + case TYPE_SHORT: + strcat( buffer, "SHORT" ) ; + break ; + + case TYPE_WORD: + strcat( buffer, "WORD" ) ; + break ; + + case TYPE_BYTE: + strcat( buffer, "BYTE" ) ; + break ; + + case TYPE_SBYTE: + strcat( buffer, "SIGNED BYTE" ) ; + break ; + + case TYPE_CHAR: + strcat( buffer, "CHAR" ) ; + break ; + + case TYPE_STRING: + strcat( buffer, "STRING" ) ; + break ; + + case TYPE_FLOAT: + strcat( buffer, "FLOAT" ) ; + break ; + + case TYPE_POINTER: + describe_type( type, from + 1 ) ; + strcat( buffer, " POINTER" ) ; + break ; + } + + return buffer ; +} + +/* --------------------------------------------------------------------------- */ + +static char * show_value( DCB_TYPEDEF type, void * data ) +{ + static char buffer[512] ; + char * newbuffer ; + int subsize; + unsigned int n ; + unsigned int count ; + + switch ( type.BaseType[0] ) + { + case TYPE_ARRAY: + count = type.Count[0]; + + type = reduce_type( type ) ; + subsize = type_size( type ) ; + if ( type.BaseType[0] == TYPE_STRUCT ) return "" ; + newbuffer = ( char * ) malloc( 512 ) ; + strcpy( newbuffer, "= (" ) ; + for ( n = 0 ; n < count ; n++ ) + { + if ( n ) strcat( newbuffer, ", " ) ; + show_value( type, data ) ; + if ( strlen( newbuffer ) + strlen( buffer ) > 30 ) + { + strcat( newbuffer, "..." ) ; + break ; + } + strcat( newbuffer, buffer + 2 ) ; + data = ( uint8_t * )data + subsize ; + } + strcat( newbuffer, ")" ) ; + strcpy( buffer, newbuffer ) ; + free( newbuffer ) ; + return buffer ; + + case TYPE_STRUCT: + return "" ; + + case TYPE_STRING: + _snprintf( buffer, 512, "= \"%s\"", string_get( *( uint32_t * )data ) ) ; + return buffer ; + + case TYPE_BYTE: + _snprintf( buffer, 512, "= %d", *( uint8_t * )data ) ; + return buffer ; + + case TYPE_SBYTE: + _snprintf( buffer, 512, "= %d", *( int8_t * )data ) ; + return buffer ; + + case TYPE_CHAR: + if ( *( uint8_t * )data >= 32 ) + _snprintf( buffer, 512, "= '%c'", *( uint8_t * )data ) ; + else + _snprintf( buffer, 512, "= '\\x%02X'", *( uint8_t * )data ) ; + return buffer ; + + case TYPE_FLOAT: + _snprintf( buffer, 512, "= %g", *( float * )data ) ; + return buffer ; + + case TYPE_WORD: + _snprintf( buffer, 512, "= %d", *( uint16_t * )data ) ; + return buffer ; + + case TYPE_DWORD: + _snprintf( buffer, 512, "= %ud", *( uint32_t * )data ) ; + return buffer ; + + case TYPE_SHORT: + _snprintf( buffer, 512, "= %d", *( int16_t * )data ) ; + return buffer ; + + case TYPE_INT: + _snprintf( buffer, 512, "= %d", *( int * )data ) ; + return buffer ; + + case TYPE_POINTER: + _snprintf( buffer, 512, "= 0x%08X", *( uint32_t * )data ) ; + return buffer ; + + default: + return "?" ; + } +} + +/* --------------------------------------------------------------------------- */ + +static void show_struct( int num, char * title, int indent, void * data ) +{ + int n, count ; + DCB_VAR * vars ; + + vars = dcb.varspace_vars[num] ; + count = dcb.varspace[num].NVars ; + + for ( n = 0 ; n < count ; n++ ) + { + show_var( vars[n], 0, data ? ( uint8_t * )data + vars[n].Offset : 0, title, indent ) ; + } +} + +/* --------------------------------------------------------------------------- */ + +static void show_var( DCB_VAR var, char * name, void * data, char * title, int indent ) +{ + char spaces[indent+1] ; + + memset( spaces, ' ', indent ) ; + spaces[indent] = 0 ; + + if ( !name ) + { + unsigned int code ; + + name = "?" ; + for ( code = 0 ; code < dcb.data.NID ; code++ ) + { + if ( dcb.id[code].Code == var.ID ) + { + name = ( char * ) dcb.id[code].Name ; + break ; + } + } + } + + if ( data ) + console_printf( "%s%s %s %s %s\n", title, spaces, describe_type( var.Type, 0 ), name, show_value( var.Type, data ) ) ; + else + console_printf( "%s%s %s %s\n", title, spaces, describe_type( var.Type, 0 ), name ) ; + + if ( var.Type.BaseType[0] == TYPE_STRUCT ) + { + show_struct( var.Type.Members, title, indent + 3, data ) ; + console_printf( "%s%s END", title, spaces ) ; + } +} + +/* --------------------------------------------------------------------------- */ + +/* Very simple tokenizer */ + +static void get_token() +{ + char * ptr ; + unsigned int n ; + + while ( isspace( *token_ptr ) ) token_ptr++ ; + + if ( !*token_ptr ) + { + token.type = NOTOKEN ; + return ; + } + + /* Numbers */ + + if ( ISNUM( *token_ptr ) ) + { + const char * ptr; + double num = 0, dec; + int base = 10; + + /* Hex/Bin/Octal numbers with the h/b/o sufix */ + ptr = token_ptr; + while ( ISNUM( *ptr ) || ( *ptr >= 'a' && *ptr <= 'f' ) || ( *ptr >= 'A' && *ptr <= 'F' ) ) ptr++; + + if ( *ptr != 'h' && *ptr != 'H' && *ptr != 'o' && *ptr != 'O' && ( ptr[-1] == 'b' || ptr[-1] == 'B' ) ) ptr--; + + if ( *ptr == 'b' || *ptr == 'B' ) base = 2; + if ( *ptr == 'h' || *ptr == 'H' ) base = 16; + if ( *ptr == 'o' || *ptr == 'O' ) base = 8; + + token.code = 0 ; /* for ints values */ + + /* Calculate the number value */ + + while ( ISNUM( *token_ptr ) || ( base > 10 && ISALNUM( *token_ptr ) ) ) + { + if ( base == 2 && *token_ptr != '0' && *token_ptr != '1' ) break; + if ( base == 8 && ( *token_ptr < '0' || *token_ptr > '7' ) ) break; + if ( base == 10 && !ISNUM( *token_ptr ) ) break; + if ( base == 16 && !ISNUM( *token_ptr ) && ( TOUPPER( *token_ptr ) < 'A' || TOUPPER( *token_ptr ) > 'F' ) ) break; + + if ( ISNUM( *token_ptr ) ) + { + num = num * base + ( *token_ptr - '0' ); + token.code = token.code * base + ( *token_ptr - '0' ); + token_ptr++; + } + if ( *token_ptr >= 'a' && *token_ptr <= 'f' && base > 10 ) + { + num = num * base + ( *token_ptr - 'a' + 10 ); + token.code = token.code * base + ( *token_ptr - 'a' + 10 ); + token_ptr++; + } + if ( *token_ptr >= 'A' && *token_ptr <= 'F' && base > 10 ) + { + num = num * base + ( *token_ptr - 'A' + 10 ); + token.code = token.code * base + ( *token_ptr - 'A' + 10 ); + token_ptr++; + } + } + token.type = NUMBER; + token.code = ( float )num; + + /* We have the integer part now - convert to int/float */ + + if ( *token_ptr == '.' && base == 10 ) + { + token_ptr++; + if ( !ISNUM( *token_ptr ) ) + token_ptr--; + else + { + dec = 1.0 / ( double )base; + while ( ISNUM( *token_ptr ) || ( base > 100 && ISALNUM( *token_ptr ) ) ) + { + if ( ISNUM( *token_ptr ) ) num = num + dec * ( *token_ptr++ - '0' ); + if ( *token_ptr >= 'a' && *token_ptr <= 'f' && base > 10 ) num = num + dec * ( *token_ptr++ - 'a' + 10 ); + if ( *token_ptr >= 'A' && *token_ptr <= 'F' && base > 10 ) num = num + dec * ( *token_ptr++ - 'A' + 10 ); + dec /= ( double )base; + } + token.type = NUMBER; + token.code = ( float )num; + } + } + + /* Skip the base sufix */ + + if ( base == 16 && ( *token_ptr == 'h' || *token_ptr == 'H' ) ) token_ptr++; + if ( base == 8 && ( *token_ptr == 'o' || *token_ptr == 'O' ) ) token_ptr++; + if ( base == 2 && ( *token_ptr == 'b' || *token_ptr == 'B' ) ) token_ptr++; + + _snprintf( token.name, sizeof( token.name ), "%g", token.code ) ; + return ; + } + + if ( *token_ptr == '"' || *token_ptr == '\'' ) /* Cadena */ + { + char c = *token_ptr++ ; + token.type = STRING ; + ptr = token.name; + while ( *token_ptr && *token_ptr != c ) *ptr++ = *token_ptr++ ; + if ( *token_ptr == c ) token_ptr++ ; + *ptr = 0 ; + return ; + } + + ptr = token.name ; + *ptr++ = TOUPPER( *token_ptr ) ; + if ( ISWORDCHAR( *token_ptr++ ) ) + { + while ( ISWORDCHAR( *token_ptr ) ) + *ptr++ = TOUPPER( *token_ptr++ ) ; + } + *ptr = 0 ; + + for ( n = 0 ; n < dcb.data.NID ; n++ ) + { + if ( strcmp( ( char * )dcb.id[n].Name, token.name ) == 0 ) + { + token.type = IDENTIFIER ; + token.code = dcb.id[n].Code ; +/* strcpy( token.name, (char *)dcb.id[n].Name ) ; */ + return ; + } + } + + token.type = OPERATOR ; +} + +/* --------------------------------------------------------------------------- */ + +static DCB_TYPEDEF reduce_type( DCB_TYPEDEF orig ) +{ + int n ; + for ( n = 0 ; n < MAX_TYPECHUNKS - 1 ; n++ ) + { + orig.BaseType[n] = orig.BaseType[n+1] ; + orig.Count[n] = orig.Count[n+1] ; + } + return orig ; +} + +/* --------------------------------------------------------------------------- */ + +static void var2const() +{ + while ( result.type == T_VARIABLE && result.var.Type.BaseType[0] == TYPE_ARRAY ) + result.var.Type = reduce_type( result.var.Type ) ; + + if ( result.type == T_VARIABLE && result.var.Type.BaseType[0] == TYPE_STRING ) + { + result.type = T_STRING ; + strncpy( result.name, string_get( *( int * )( result.data ) ), 127 ) ; + result.name[127] = 0 ; + } + + if ( result.type == T_VARIABLE && result.var.Type.BaseType[0] == TYPE_FLOAT ) + { + result.type = T_CONSTANT ; + result.value = *( float * )( result.data ) ; + } + + if ( result.type == T_VARIABLE && ( result.var.Type.BaseType[0] == TYPE_DWORD || result.var.Type.BaseType[0] == TYPE_INT ) ) + { + result.type = T_CONSTANT ; + result.value = *( int * )( result.data ) ; + } + + if ( result.type == T_VARIABLE && ( result.var.Type.BaseType[0] == TYPE_WORD || result.var.Type.BaseType[0] == TYPE_SHORT ) ) + { + result.type = T_CONSTANT ; + result.value = *( int16_t * )( result.data ) ; + } + + if ( result.type == T_VARIABLE && ( result.var.Type.BaseType[0] == TYPE_BYTE || result.var.Type.BaseType[0] == TYPE_SBYTE ) ) + { + result.type = T_CONSTANT ; + result.value = *( int8_t * )( result.data ) ; + } + + if ( result.type == T_VARIABLE && result.var.Type.BaseType[0] == TYPE_CHAR ) + { + result.type = T_STRING ; + if ( *( uint8_t * )result.data >= 32 ) + _snprintf( result.name, sizeof( result.name ), "%c", *( uint8_t * )result.data ) ; + else + _snprintf( result.name, sizeof( result.name ), "\\x%02X", *( uint8_t * )result.data ) ; + } +} + +/* --------------------------------------------------------------------------- */ + +static int type_size( DCB_TYPEDEF orig ) +{ + unsigned int n, total ; + + switch ( orig.BaseType[0] ) + { + case TYPE_ARRAY: + return orig.Count[0] * type_size( reduce_type( orig ) ) ; + + case TYPE_POINTER: + case TYPE_STRING: + case TYPE_DWORD: + case TYPE_FLOAT: + case TYPE_INT: + return 4 ; + + case TYPE_WORD: + case TYPE_SHORT: + return 2 ; + + case TYPE_BYTE: + case TYPE_SBYTE: + case TYPE_CHAR: + return 1 ; + + case TYPE_STRUCT: + total = 0 ; + for ( n = 0 ; n < dcb.varspace[orig.Members].NVars ; n++ ) + total += type_size( dcb.varspace_vars[orig.Members][n].Type ) ; + return total ; + + default: + return 0 ; + } +} + +/* --------------------------------------------------------------------------- */ + +static void eval_local( DCB_PROC * proc, INSTANCE * i ) +{ + unsigned int n ; + + for ( n = 0 ; n < dcb.data.NLocVars ; n++ ) + { + if ( dcb.locvar[n].ID == token.code ) + { + strcpy( result.name, token.name ) ; + result.type = T_VARIABLE ; + result.var = dcb.locvar[n] ; + result.data = ( uint8_t * )i->locdata + dcb.locvar[n].Offset ; + get_token() ; + return ; + } + } + + for ( n = 0 ; n < proc->data.NPriVars ; n++ ) + { + if ( proc->privar[n].ID == token.code ) + { + strcpy( result.name, token.name ) ; + result.type = T_VARIABLE ; + result.var = proc->privar[n] ; + result.data = ( uint8_t * )i->pridata + proc->privar[n].Offset ; + get_token() ; + return ; + } + } + + for ( n = 0 ; n < proc->data.NPubVars ; n++ ) + { + if ( proc->pubvar[n].ID == token.code ) + { + strcpy( result.name, token.name ) ; + result.type = T_VARIABLE ; + result.var = proc->pubvar[n] ; + result.data = ( uint8_t * )i->pubdata + proc->pubvar[n].Offset ; + get_token() ; + return ; + } + } + + console_printf( "¬02Local or private or public variable not found¬07" ) ; + result.type = T_ERROR ; +} + +/* --------------------------------------------------------------------------- */ + +static void eval_immediate() +{ + unsigned int n ; + + if ( token.type == NUMBER ) + { + result.type = T_CONSTANT ; + result.value = token.code ; + get_token() ; + return ; + } + + if ( token.type == STRING ) + { + result.type = T_STRING ; + _snprintf( result.name, sizeof( result.name ), "%s", token.name ) ; + get_token() ; + return ; + } + + if ( token.type != IDENTIFIER ) + { + console_printf( "¬02Not a valid expression¬07" ) ; + result.type = T_ERROR ; + return ; + } + + if ( token.name[0] == '(' ) + { + get_token() ; + eval_subexpression() ; + if ( token.name[0] != ')' ) + { + console_printf( "¬02Unbalanced parens¬07" ) ; + result.type = T_ERROR ; + return ; + } + get_token() ; + return ; + } + + if ( token.name[0] == '-' ) + { + get_token() ; + eval_immediate() ; + var2const() ; + if ( result.type != T_CONSTANT ) + { + console_printf( "¬02Operand is not a number¬07" ) ; + result.type = T_ERROR ; + return ; + } + result.value = -result.value ; + _snprintf( result.name, sizeof( result.name ), "%g", result.value ) ; + return ; + } + + for ( n = 0 ; n < dcb.data.NGloVars ; n++ ) + { + if ( dcb.glovar[n].ID == token.code ) + { + strcpy( result.name, token.name ) ; + result.type = T_VARIABLE ; + result.var = dcb.glovar[n] ; + result.data = ( uint8_t * )globaldata + dcb.glovar[n].Offset ; + + get_token() ; + return ; + } + } + + if ( strcmp( token.name, "MAIN" ) == 0 ) + token.code = dcb.proc[0].data.ID ; + + for ( n = 0 ; n < dcb.data.NProcs ; n++ ) + { + if ( dcb.proc[n].data.ID == token.code ) + { + INSTANCE * i = first_instance ; + + while ( i ) + { + if ( i->proc->type == ( int )n ) break ; + i = i->next ; + } + if ( !i ) + { + console_printf( "¬02No instance of process %s is active¬07", token.name ) ; + result.type = T_ERROR ; + return ; + } + + get_token() ; + if ( token.name[0] != '.' ) + { + console_printf( "¬02Invalid use of a process name¬07" ) ; + result.type = T_ERROR ; + return ; + } + get_token() ; + eval_local( &dcb.proc[n], i ) ; + return ; + } + } + + console_printf( "¬02Variable does not exist (%s)¬07", token.name ) ; + result.type = T_ERROR ; + return ; +} + +/* --------------------------------------------------------------------------- */ + +static void eval_value() +{ + eval_immediate() ; + if ( result.type == T_ERROR ) return ; + + for ( ;; ) + { + if ( token.name[0] == '.' ) + { + DCB_VARSPACE * v ; + DCB_VAR * var ; + unsigned int n ; + + var2const() ; + if ( result.type == T_CONSTANT ) + { + INSTANCE * i ; + i = instance_get(( int )result.value ) ; + if ( !i ) + { + result.type = T_ERROR ; + console_printf( "¬02Instance %d does not exist¬07", ( int )result.value ) ; + return ; + } + get_token() ; + eval_local( &dcb.proc[i->proc->type], i ) ; + continue; + } + + if ( result.type != T_VARIABLE + || result.var.Type.BaseType[0] != TYPE_STRUCT ) + { + console_printf( "¬02%s is not an struct¬07", result.name ); + result.type = T_ERROR ; + return ; + } + get_token() ; + if ( token.type != IDENTIFIER ) + { + console_printf( "¬02%s is not a member¬07", token.name ) ; + result.type = T_ERROR ; + return ; + } + + v = &dcb.varspace[result.var.Type.Members] ; + var = dcb.varspace_vars[result.var.Type.Members] ; + for ( n = 0 ; n < v->NVars ; n++ ) + { + if ( var[n].ID == token.code ) + break ; + } + if ( n == v->NVars ) + { + console_printf( "¬02%s is not a member¬07", token.name ) ; + result.type = T_ERROR ; + return ; + } + + result.type = T_VARIABLE ; + result.var = var[n] ; + result.data = ( uint8_t * )result.data + var[n].Offset ; + strcat( result.name, "." ) ; + strcat( result.name, token.name ) ; + get_token() ; + continue ; + } + + if ( token.name[0] == '[' ) + { + DCB_VAR i = result.var ; + void * i_data = result.data ; + char name[256] ; + + if ( result.type != T_VARIABLE || result.var.Type.BaseType[0] != TYPE_ARRAY ) + { + console_printf( "¬02%s is not an array¬07", result.name ) ; + result.type = T_ERROR ; + return ; + } + + strcpy( name, result.name ) ; + get_token() ; + eval_subexpression() ; + if ( result.type == T_ERROR ) return ; + if ( token.name[0] == ']' ) get_token() ; + var2const() ; + + if ( result.type != T_CONSTANT ) + { + console_printf( "¬02%s is not an integer¬07", result.name ) ; + result.type = T_ERROR ; + return ; + } + if ( result.value < 0 ) + { + console_printf( "¬02Index (%d) less than zero¬07", result.value ) ; + result.type = T_ERROR ; + return ; + } + if ( result.value >= i.Type.Count[0] ) + { + console_printf( "¬02Index (%d) out of bounds¬07", result.value ) ; + result.type = T_ERROR ; + return ; + } + + result.type = T_VARIABLE ; + result.var = i ; + result.var.Type = reduce_type( i.Type ) ; + result.data = ( uint8_t * )i_data + ( int )result.value * type_size( result.var.Type ) ; + _snprintf( result.name, sizeof( result.name ), "%s[%d]", name, ( int )result.value ) ; + continue ; + } + + break ; + } +} + +/* --------------------------------------------------------------------------- */ + +static void eval_factor() +{ + double base = 1 ; + int op = 0 ; + + for ( ;; ) + { + eval_value() ; + if ( result.type == T_ERROR ) return ; + if ( !strchr( "*/%", token.name[0] ) && !op ) return ; + var2const() ; + if ( result.type != T_CONSTANT ) + { + result.type = T_ERROR ; + console_printf( "¬02Operand is not a number¬07" ) ; + return ; + } + if ( !op ) op = 1 ; + if ( op > 1 && !result.value ) + { + result.type = T_ERROR ; + console_printf( "¬02Divide by zero¬07" ) ; + return ; + } + if ( op == 1 ) base *= result.value ; + if ( op == 2 ) base /= result.value ; + if ( op == 3 ) base = ( int )base % ( int )result.value ; + if ( !strchr( "*/%", token.name[0] ) ) + { + result.type = T_CONSTANT ; + result.value = base ; + _snprintf( result.name, sizeof( result.name ), "%g", base ) ; + return ; + } + op = ( token.name[0] == '*' ) ? 1 : ( token.name[0] == '/' ) ? 2 : 3 ; + get_token() ; + } +} + +/* --------------------------------------------------------------------------- */ + +static void eval_subexpression() +{ + double base = 0 ; + int op = 0 ; + + for ( ;; ) + { + eval_factor() ; + if ( result.type == T_ERROR ) return ; + if ( token.name[0] != '+' && token.name[0] != '-' && !op ) return ; + var2const() ; + if ( result.type != T_CONSTANT ) + { + result.type = T_ERROR ; + console_printf( "¬02Operand is not a number¬07" ) ; + return ; + } + if ( !op ) op = 1 ; + base += op * result.value ; + if ( token.name[0] != '+' && token.name[0] != '-' ) + { + result.type = T_CONSTANT ; + result.value = base ; + _snprintf( result.name, sizeof( result.name ), "%g", base ) ; + return ; + } + op = ( token.name[0] == '+' ) ? 1 : -1 ; + get_token() ; + } +} + +/* --------------------------------------------------------------------------- */ + +static char * eval_expression( const char * here, int interactive ) +{ + static char buffer[1024]; + static char part[1024]; + + while ( *here == ' ' ) here++; + + token_ptr = here ; + get_token() ; + eval_subexpression() ; + + if ( token.type != NOTOKEN && token.name[0] != ',' && token.name[0] != '=' ) + { + if ( result.type != T_ERROR ) + { + console_printf( "¬02Invalid expression¬07" ); + result.type = T_ERROR; + } + return 0; + } + + memset( part, 0, sizeof( buffer ) ); + strncpy( part, here, token_ptr - here - (( token.type != NOTOKEN ) ? 1 : 0 ) ); + + if ( result.type == T_CONSTANT ) + { + _snprintf( buffer, 1024, "%s = %g", part, result.value ); + if ( interactive ) console_printf( "¬07%s", buffer ) ; + } + else if ( result.type == T_STRING ) + { + if ( interactive ) console_printf( "¬07%s = \"%s\"", part, result.name ) ; + } + else if ( result.type == T_VARIABLE ) + { + lvalue = result ; + + if ( token.name[0] == '=' ) + { + if ( lvalue.type != T_VARIABLE ) + { + strcpy( buffer, "Not an lvalue" ) ; + if ( interactive ) console_printf( "¬02%s¬07", buffer ) ; + return buffer ; + } + get_token() ; + eval_subexpression() ; + if ( result.type == T_ERROR ) return "" ; + var2const() ; + + if (( lvalue.var.Type.BaseType[0] == TYPE_DWORD || lvalue.var.Type.BaseType[0] == TYPE_INT ) && result.type == T_CONSTANT ) + *( int * )( lvalue.data ) = ( int )result.value ; + else if (( lvalue.var.Type.BaseType[0] == TYPE_WORD || lvalue.var.Type.BaseType[0] == TYPE_SHORT ) && result.type == T_CONSTANT ) + *( uint16_t * )( lvalue.data ) = ( uint16_t )result.value ; + else if (( lvalue.var.Type.BaseType[0] == TYPE_BYTE || lvalue.var.Type.BaseType[0] == TYPE_SBYTE ) && result.type == T_CONSTANT ) + *( uint8_t * )( lvalue.data ) = ( uint8_t )result.value ; + else if ( lvalue.var.Type.BaseType[0] == TYPE_CHAR && result.type == T_STRING ) + { + if ( *result.name == '\\' && *( result.name + 1 ) == 'x' ) + *( uint8_t * )( lvalue.data ) = ( uint8_t )strtol( result.name + 2, NULL, 16 ); + else + *( uint8_t * )( lvalue.data ) = ( uint8_t ) * ( result.name ); + } + else if ( lvalue.var.Type.BaseType[0] == TYPE_FLOAT && result.type == T_CONSTANT ) + *( float * )( lvalue.data ) = ( float )result.value ; + else if ( lvalue.var.Type.BaseType[0] == TYPE_STRING && result.type == T_STRING ) + { + string_discard( *( uint32_t * ) lvalue.data ) ; + *( uint32_t * )( lvalue.data ) = string_new( result.name ) ; + string_use( *( uint32_t * ) lvalue.data ) ; + } + else + { + strcpy( buffer, "Invalid assignation" ) ; + if ( interactive ) console_printf( "¬02%s¬07", buffer ) ; + return buffer ; + } + } + + if ( interactive ) + { + show_var( lvalue.var, lvalue.name, lvalue.data, "", 0 ) ; + } + else + { + strcpy( buffer, part ) ; + strcat( buffer, " " ) ; + strcat( buffer, show_value( lvalue.var.Type, lvalue.data ) ); + } + } + + if ( token.name[0] == ',' ) + { + char * temporary = strdup( buffer ); + int size = strlen( temporary ); + + if ( eval_expression( token_ptr, interactive ) == 0 ) + { + free( temporary ); + return 0; + } + if ( strlen( buffer ) + size < 1020 && !interactive ) + { + memmove( buffer + size + 2, buffer, strlen( buffer ) + 1 ); + memcpy( buffer, temporary, size ); + memcpy( buffer + size, ", ", 2 ); + } + free( temporary ); + return buffer; + } + + return buffer; +} + +/* --------------------------------------------------------------------------- */ + +static void console_instance_dump( INSTANCE * father, int indent ) +{ + INSTANCE * i, * next ; + PROCDEF * proc ; + DCB_PROC * dcbproc ; + char buffer[128] ; + char line[128] ; + int n, nid, bigbro, son ; + + i = father ; + if ( !father ) i = first_instance ; + + proc = i->proc ; + dcbproc = &dcb.proc[proc->type] ; + + nid = LOCDWORD( mod_debug, i, PROCESS_ID ) ; + if ( dcb.data.NSourceFiles && dcbproc->data.ID ) + sprintf( buffer, "%s", getid_name( dcbproc->data.ID ) ) ; + else + sprintf( buffer, "%s", ( proc->type == 0 ) ? "MAIN" : "PROC" ) ; + + line[0] = 0 ; + for ( n = 0 ; n < indent - 1; n++ ) strcat( line, " \x03 " ); + + sprintf( line + strlen( line ), " \x01\x02 %-12s ", buffer ) ; + + n = strlen( line ) % 4 ; + while ( n && ( n++ < 4 ) ) strcat( line, " " ) ; + sprintf( line + strlen( line ), "%7d", nid ) ; + + if ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) strcat( line, "[W]" ); + switch ( LOCDWORD( mod_debug, i, STATUS ) & ~STATUS_WAITING_MASK ) + { + case STATUS_DEAD : strcat( line, "[D]" ) ; break ; + case STATUS_KILLED : strcat( line, "[K]" ) ; break ; + case STATUS_RUNNING : strcat( line, " " ) ; break ; + case STATUS_SLEEPING : strcat( line, "[S]" ) ; break ; + case STATUS_FROZEN : strcat( line, "[F]" ) ; break ; + } + + console_printf( "¬07%s", line ) ; + + if ( !( son = LOCDWORD( mod_debug, i, SON ) ) ) return ; + + next = instance_get( son ) ; + if ( !next ) console_printf( "¬02\12**PANIC**\7 SON %d does not exist¬07", son ) ; + + i = next ; + + while ( i ) + { + proc = i->proc ; + dcbproc = &dcb.proc[proc->type] ; + + if ( LOCDWORD( mod_debug, i, SON ) ) + { + console_instance_dump( i, indent + 1 ) ; + } + else + { + nid = LOCDWORD( mod_debug, i, PROCESS_ID ) ; + if ( dcb.data.NSourceFiles && dcbproc->data.ID ) + { + sprintf( buffer, "%s", getid_name( dcbproc->data.ID ) ) ; + } + else + { + sprintf( buffer, "%s", ( proc->type == 0 ) ? "MAIN" : "PROC" ) ; + } + + line[0] = 0 ; + for ( n = 0 ; n < indent; n++ ) strcat( line, " \x03 " ); + + sprintf( line + strlen( line ), " \x01\x02 %-12s ", buffer ) ; + + n = strlen( line ) % 4 ; + while ( n && ( n++ < 4 ) ) strcat( line, " " ) ; + sprintf( line + strlen( line ), "%7d", nid ) ; + + if ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) strcat( line, "[W]" ); + switch ( LOCDWORD( mod_debug, i, STATUS ) & ~STATUS_WAITING_MASK ) + { + case STATUS_DEAD : strcat( line, "[D]" ) ; break ; + case STATUS_KILLED : strcat( line, "[K]" ) ; break ; + case STATUS_RUNNING : strcat( line, " " ) ; break ; + case STATUS_SLEEPING : strcat( line, "[S]" ) ; break ; + case STATUS_FROZEN : strcat( line, "[F]" ) ; break ; + } + console_printf( "¬07%s", line ) ; + } + + if ( ( bigbro = LOCDWORD( mod_debug, i, BIGBRO ) ) ) + { + next = instance_get( bigbro ) ; + if ( !next ) console_printf( "¬02\12**PANIC**\7 BIGBRO %d does not exist¬07", bigbro ) ; + i = next ; + } + else + break ; + } +} + +/* --------------------------------------------------------------------------- */ +/* Muestra el árbol de instancias */ + +static void console_instance_dump_all() +{ + INSTANCE * i ; + int father; + + console_printf( "¬04INSTANCES TREE¬07\n\n" ); + + for ( i = first_instance ; i ; i = i->next ) + { + if ( !( father = LOCDWORD( mod_debug, i, FATHER )) || !instance_get( father ) ) + { + console_instance_dump( i, 1 ) ; + } + } + + console_printf( "\n" ) ; +} + +/* --------------------------------------------------------------------------- */ + +static void console_instance_dump_all_brief() +{ + INSTANCE * i ; + char status[30] ; + int father; + + console_printf( "¬04INSTANCES BRIEF LIST¬07\n\n" ); + console_printf( "¬04Id Father Status Name¬07\n" ); + + + for ( i = first_instance ; i ; i = i->next ) + { + status[0] = '\0'; + if ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) + strcpy( status, "¬08wait¬07+" ); + + switch ( LOCDWORD( mod_debug, i, STATUS ) & ~STATUS_WAITING_MASK ) + { + case STATUS_DEAD : strcat( status, "¬06dead¬07 " ) ; break ; + case STATUS_KILLED : strcat( status, "¬02killed¬07 " ) ; break ; + case STATUS_SLEEPING : strcat( status, "¬09sleeping¬07" ) ; break ; + case STATUS_FROZEN : strcat( status, "¬05frozen¬07 " ) ; break ; + case STATUS_RUNNING : strcat( status, "¬12running¬07 " ) ; break ; + } + + if ( !( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) ) + strcat( status, " " ); + + + father = LOCDWORD( mod_debug, i, FATHER ); + + console_printf( "%-10d %s%-10d¬07 %s ¬15%s¬07\n", + LOCDWORD( mod_debug, i, PROCESS_ID ), + instance_get( father ) ? "" : "¬02", + father, + status, + ( dcb.data.NSourceFiles && dcb.proc[i->proc->type].data.ID ) ? getid_name( dcb.proc[i->proc->type].data.ID ) : (( i->proc->type == 0 ) ? "Main" : "proc" ) + ) ; + } + + console_printf( "\n" ) ; +} + +/* --------------------------------------------------------------------------- */ + +static int console_list_current = 0; +static int console_list_y_pos[2] = { 0 }; +static int console_list_x_pos[2] = { 0 }; +static int console_list_y_max[2] = { 0 }; +static INSTANCE * console_list_current_instance = NULL; + +#define WINDOW_LIST_COLS 40 + +static void show_list_window() +{ + int n; + int pos = console_list_y_pos[console_list_current] > console_y / CHARHEIGHT - 2 ? console_list_y_pos[console_list_current] - ( console_y / CHARHEIGHT - 2 ) : 0 ; + int x = ( scrbitmap->width - console_columns * CHARWIDTH ) / 2 + ( console_columns - WINDOW_LIST_COLS ) * CHARWIDTH ; + int y = 0; + + systext_color( 0x000000, 0xC0C0C0 ) ; + switch ( console_list_current ) + { + case 0: + console_list_y_max[0] = procdef_count ; + systext_puts( scrbitmap, x, y, "¬00¬21PROCESS TYPES", WINDOW_LIST_COLS ); + for ( n = 0 ; pos + n < procdef_count && y < console_y - CHARHEIGHT ; n++ ) + { + y += CHARHEIGHT; + if ( console_list_y_pos[console_list_current] == pos + n ) + { + if ( procs[pos + n].breakpoint ) + systext_puts( scrbitmap, x, y, "¬15¬19", 1 ); + else + systext_puts( scrbitmap, x, y, "¬15¬17", 1 ); + } + else + { + if ( procs[pos + n].breakpoint ) + systext_puts( scrbitmap, x, y, "¬15¬18", 1 ); + else + systext_puts( scrbitmap, x, y, "¬00¬23", 1 ); + } + if ( console_list_x_pos[console_list_current] < strlen( procs[pos + n].name ) ) + systext_puts( scrbitmap, x, y, &procs[pos + n].name[console_list_x_pos[console_list_current]], WINDOW_LIST_COLS ); + else + systext_puts( scrbitmap, x, y, "", WINDOW_LIST_COLS ); + } + break; + + case 1: + { + INSTANCE * i ; + int c ; + char line[128]; + char status[10]; + + console_list_y_max[1] = 0 ; + + systext_puts( scrbitmap, x, y, "¬00¬21INSTANCES", WINDOW_LIST_COLS ); + + for ( c = 0, n = 0, i = first_instance ; i ; i = i->next, c++ ) + { + console_list_y_max[1]++ ; + + if ( c < pos || y >= console_y - CHARHEIGHT ) continue ; + y += CHARHEIGHT; + + if ( console_list_y_pos[console_list_current] == pos + n ) + { + console_list_current_instance = i; + + if ( i->breakpoint ) + systext_puts( scrbitmap, x, y, "¬15¬19", 1 ); + else + systext_puts( scrbitmap, x, y, "¬15¬17", 1 ); + } + else + { + if ( i->breakpoint ) + systext_puts( scrbitmap, x, y, "¬15¬18", 1 ); + else + systext_puts( scrbitmap, x, y, "¬00¬23", 1 ); + } + + status[0] = '\0'; + if ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) strcat( status, "[W]" ); + switch ( LOCDWORD( mod_debug, i, STATUS ) & ~STATUS_WAITING_MASK ) + { + case STATUS_DEAD : strcat( status, "[D]" ) ; break ; + case STATUS_KILLED : strcat( status, "[K]" ) ; break ; + case STATUS_SLEEPING : strcat( status, "[S]" ) ; break ; + case STATUS_FROZEN : strcat( status, "[F]" ) ; break ; + case STATUS_RUNNING : strcat( status, "[R]" ) ; break ; + } + + sprintf( line, "%d %-6.6s %s", + LOCDWORD( mod_debug, i, PROCESS_ID ), + status, + ( dcb.data.NSourceFiles && dcb.proc[i->proc->type].data.ID ) ? getid_name( dcb.proc[i->proc->type].data.ID ) : (( i->proc->type == 0 ) ? "Main" : "proc" ) + ); + if ( console_list_x_pos[console_list_current] < strlen( line ) ) + systext_puts( scrbitmap, x, y, &line[console_list_x_pos[console_list_current]], WINDOW_LIST_COLS ); + else + systext_puts( scrbitmap, x, y, "", WINDOW_LIST_COLS ); + + n++ ; + } + } + break; + } +} + +/* --------------------------------------------------------------------------- */ + +static INSTANCE * findproc( INSTANCE * last, char * action, char * ptr ) +{ + INSTANCE * i = NULL; + char * aptr; + int procno = 0; + int n; + + if ( *ptr ) + { + if ( *ptr >= '0' && *ptr <= '9' ) + { + if ( last ) return NULL; + procno = atoi( ptr ); + for ( i = first_instance ; i ; i = i->next ) + if ( LOCDWORD( mod_debug, i, PROCESS_ID ) == procno ) + break; + if ( !i ) + { + console_printf( "¬02Instance %d does not exist¬07", procno ); + return NULL; + } + } + else + { + aptr = action; + while ( ISWORDCHAR( *ptr ) ) + { + *aptr++ = TOUPPER( *ptr ); ptr++; + } + *aptr = 0; + + for ( n = 0 ; n < ( int )dcb.data.NID ; n++ ) + if ( strcmp( ( char * )dcb.id[n].Name, action ) == 0 ) + break; + for ( procno = 0 ; procno < ( int )dcb.data.NProcs ; procno++ ) + if ( dcb.proc[procno].data.ID == dcb.id[n].Code ) + break; + if ( procno == ( int )dcb.data.NProcs ) + { + console_printf( "¬02Unknown process %s¬07", action ); + return NULL; + } + for ( i = last ? last->next : first_instance ; i ; i = i->next ) + if ( i->proc->type == procno ) + break; + if ( !i && !last ) + { + console_printf( "¬02No instance of process %s created¬07", action ); + return NULL; + } + } + } + + return ( i ); +} + +/* --------------------------------------------------------------------------- */ + +static void console_do( const char * command ) +{ + char * ptr, * aptr ; + char action[256] ; + unsigned int var ; + int n, procno = 0; + PROCDEF * p = NULL ; + INSTANCE * i = NULL ; + + ptr = ( char * ) command ; + while ( *ptr && *ptr != ' ' ) ptr++ ; + strncpy( action, command, ptr - command ) ; + action[ptr - command] = 0 ; + while ( *ptr == ' ' ) ptr++ ; + aptr = action ; + while ( *aptr ) + { + *aptr = TOUPPER( *aptr ); + aptr++; + } + + /* Comandos */ + + if ( strcmp( action, "HELP" ) == 0 ) + { + console_printf( HELPTXT ); + return; + } + + if ( strcmp( action, "GO" ) == 0 ) + { + SDL_EnableKeyRepeat( 0, 0 ); + debug_mode = 0; + debug_on_frame = 0; + force_debug = 0; + debug_next = 0; + break_on_next_proc = 0; + console_showing = 0 ; + return ; + } + + if ( strcmp( action, "NEXTFRAME" ) == 0 ) + { + SDL_EnableKeyRepeat( 0, 0 ); + break_on_next_proc = 0; + debug_mode = 0; + debug_on_frame = 1; + force_debug = 0; + debug_next = 0; + break_on_next_proc = 0; + return ; + } + + if ( strcmp( action, "NEXTPROC" ) == 0 ) + { + SDL_EnableKeyRepeat( 0, 0 ); + debug_mode = 0; + debug_on_frame = 0; + force_debug = 0; + debug_next = 0; + break_on_next_proc = 1; + return ; + } + + if ( strcmp( action, "TRACE" ) == 0 ) + { + SDL_EnableKeyRepeat( 0, 0 ); + debug_mode = 0; + debug_on_frame = 0; + force_debug = 0; + debug_next = 1; + break_on_next_proc = 0; + return ; + } + + if ( strcmp( action, "BREAK" ) == 0 ) + { + if ( *ptr ) + { + if ( *ptr >= '0' && *ptr <= '9' ) + { + procno = atoi( ptr ); + for ( i = first_instance ; i ; i = i->next ) + if ( LOCDWORD( mod_debug, i, PROCESS_ID ) == procno ) + break; + if ( !i ) + { + console_printf( "¬02Instance %d does not exist¬07", procno ); + } + else + { + i->breakpoint = 1; + console_printf( "¬07OK" ); + } + } + else + { + aptr = action; + while ( ISWORDCHAR( *ptr ) ) + { + *aptr++ = TOUPPER( *ptr ); ptr++; + } + *aptr = 0; + + if ( *action ) + { + p = procdef_get_by_name( action ); + if ( !p ) + { + console_printf( "¬02Process type %d does not exist¬07", action ); + } + else + { + p->breakpoint = 1; + console_printf( "¬07OK" ); + } + } + } + } + else + { + int f = 0; + for ( n = 0 ; n < procdef_count; n++ ) + { + if ( procs[n].breakpoint ) + { + if ( !f ) + { + console_printf( "¬04PROCESS TYPE BREAKPOINTS¬07\n\n" ); + f = 1; + } + console_printf( "¬07%s\n", procs[n].name ); + } + } + if ( f ) + console_printf( "\n" ); + + f = 0; + for ( i = first_instance ; i ; i = i->next ) + { + if ( i->breakpoint ) + { + if ( !f ) + { + console_printf( "¬04PROCESS BREAKPOINTS¬07\n\n" ); + f = 1; + } + console_printf( "¬07%d", LOCDWORD( mod_debug, i, PROCESS_ID ) ); + } + } + if ( f ) console_printf( "\n" ); + } + return ; + } + + if ( strcmp( action, "BREAKALL" ) == 0 ) + { + for ( i = first_instance ; i ; i = i->next ) i->breakpoint = 1; + console_printf( "¬07OK" ); + return ; + } + + if ( strcmp( action, "BREAKALLTYPES" ) == 0 ) + { + for ( n = 0 ; n < procdef_count; n++ ) procs[n].breakpoint = 1; + console_printf( "¬07OK" ); + return ; + } + + if ( strcmp( action, "DELETEALL" ) == 0 ) + { + for ( i = first_instance ; i ; i = i->next ) i->breakpoint = 0; + console_printf( "¬07OK" ); + return ; + } + + if ( strcmp( action, "DELETEALLTYPES" ) == 0 ) + { + for ( n = 0 ; n < procdef_count; n++ ) procs[n].breakpoint = 0; + console_printf( "¬07OK" ); + return ; + } + + if ( strcmp( action, "DELETE" ) == 0 ) + { + if ( *ptr ) + { + if ( *ptr >= '0' && *ptr <= '9' ) + { + procno = atoi( ptr ); + for ( i = first_instance ; i ; i = i->next ) + if ( LOCDWORD( mod_debug, i, PROCESS_ID ) == procno ) + break; + if ( !i ) + { + console_printf( "¬02Instance %d does not exist¬07", procno ); + } + else + { + i->breakpoint = 0; + console_printf( "¬07OK" ); + } + } + else + { + aptr = action; + while ( ISWORDCHAR( *ptr ) ) + { + *aptr++ = TOUPPER( *ptr ); ptr++; + } + *aptr = 0; + + p = procdef_get_by_name( action ); + if ( !p ) + { + console_printf( "¬02Process type %d does not exist¬07", procno ); + } + else + { + p->breakpoint = 0; + console_printf( "¬07OK" ); + } + } + } + return ; + } + + if ( strcmp( action, "STRINGS" ) == 0 ) + { + string_dump(console_printf) ; + return ; + } + + if ( strcmp( action, "INSTANCES" ) == 0 ) + { + console_instance_dump_all() ; + return ; + } + + if ( strcmp( action, "GLOBALS" ) == 0 ) + { + for ( var = 0 ; var < dcb.data.NGloVars ; var++ ) + { + DCB_VAR * v = &dcb.glovar[var] ; + show_var( *v, 0, ( uint8_t * )globaldata + v->Offset, "[GLO]", 0 ) ; + } + console_printf( "\n" ); + return ; + } + + if ( + strcmp( action, "LOCALS" ) == 0 || + strcmp( action, "PRIVATES" ) == 0 || + strcmp( action, "PUBLICS" ) == 0 ) + { + int show_locals = action[0] == 'L'; + int show_public = action[0] == 'P' && action[1] == 'U' ; + + i = findproc( NULL, action, ptr ); + + if ( show_locals ) + { + for ( var = 0 ; var < dcb.data.NLocVars ; var++ ) + { + DCB_VAR * v = &dcb.locvar[var] ; + show_var( *v, 0, i ? ( char* )i->locdata + v->Offset : 0, "[LOC]", 0 ) ; + } + } + else if ( show_public ) + { + if ( !i ) + { + console_printf( "¬07Use: PUBLICS process" ); + return; + } + for ( var = 0 ; var < dcb.proc[i->proc->type].data.NPubVars ; var++ ) + { + DCB_VAR * v = &dcb.proc[i->proc->type].pubvar[var] ; + + /* Unnamed private vars are temporary params and loop + counters, and are ignored by this command */ + if (( int )v->ID >= 0 ) + { + show_var( *v, 0, ( char* )i->pubdata + v->Offset, "[PUB]", 0 ) ; + } + } + } + else + { + if ( !i ) + { + console_printf( "¬07Use: PRIVATES process" ); + return; + } + for ( var = 0 ; var < dcb.proc[i->proc->type].data.NPriVars ; var++ ) + { + DCB_VAR * v = &dcb.proc[i->proc->type].privar[var] ; + + /* Unnamed private vars are temporary params and loop + counters, and are ignored by this command */ + if (( int )v->ID >= 0 ) + { + show_var( *v, 0, ( char* )i->pridata + v->Offset, "[PRI]", 0 ) ; + } + } + } + console_printf( "\n" ); + return ; + } + + if ( strcmp( action, "SHOW" ) == 0 ) + { + if ( *ptr ) + { + char * res = eval_expression( ptr, 0 ); + + if ( !res || result.type == T_STRING ) + { + console_printf( "¬02Invalid argument¬07" ); + return ; + } + + for ( n = 0; n < MAX_EXPRESSIONS; n++ ) + { + if ( show_expression[n] && !strcmp( show_expression[n], ptr ) ) + { + console_printf( "¬02Already exists¬07" ); + return ; + } + } + + for ( n = 0; n < MAX_EXPRESSIONS; n++ ) + { + if ( !show_expression[n] ) + { + show_expression[n] = strdup( ptr ); + show_expression_count++; + console_printf( "¬07OK" ); + return; + } + } + + console_printf( "¬02No more expressions are possibles¬07" ); + } + else + { + int nn; + for ( n = 0; n < MAX_EXPRESSIONS; n++ ) + { + if ( show_expression[n] ) + { + console_printf( "%d: %s", n + 1, show_expression[n] ); + nn++; + } + } + + if ( !nn ) console_printf( "¬02No expressions availables¬07" ); + } + + return; + } + + + if ( strcmp( action, "SHOWDEL" ) == 0 ) + { + if ( *ptr ) + { + char * p = ptr; + + while( ISNUM( *p ) ) p++; + + if ( ISNUM( *ptr ) ) + { + int pos = atol( ptr ) - 1; + if ( pos >= 0 && pos < MAX_EXPRESSIONS && show_expression[pos] ) + { + free( show_expression[pos] ); + show_expression[pos] = NULL; + show_expression_count--; + console_printf( "¬07OK" ); + return; + } + } + } + + console_printf( "¬02Invalid argument¬07" ); + + return; + } + + if ( strcmp( action, "SHOWDELALL" ) == 0 ) + { + for ( n = 0; n < MAX_EXPRESSIONS; n++ ) + { + if ( show_expression[n] ) + { + free( show_expression[n] ); + show_expression[n] = NULL; + show_expression_count--; + } + } + console_printf( "¬07OK" ); + return; + } + + if ( strcmp( action, "VARS" ) == 0 ) + { + for ( var = 0 ; var < N_CONSOLE_VARS ; var++ ) + { + switch ( console_vars[var].type ) + { + case CON_DWORD: + console_printf( "¬07%s = %d\n", console_vars[var].name, *( int * )console_vars[var].value ) ; + break; + + case CON_DWORD_HEX: + console_printf( "¬07%s = %08Xh\n", console_vars[var].name, *( int * )console_vars[var].value ) ; + break; + } + } + console_printf( "\n" ); + return ; + } + + if ( strcmp( action, "RUN" ) == 0 ) + { + if ( *ptr ) + { + aptr = action; + while ( ISWORDCHAR( *ptr ) ) + { + *aptr++ = TOUPPER( *ptr ); ptr++; + } + *aptr = 0; + + if ( *action ) + { + int i; + INSTANCE * inst ; + p = procdef_get_by_name( action ); + if ( p ) + { + token_ptr = ptr ; + console_printf( "¬07%s", ptr ); + inst = instance_new( p, NULL ); + + for ( i = 0; i < p->params; i++ ) + { + int type = dcb.proc[p->type].privar[i].Type.BaseType[0]; + get_token() ; + eval_subexpression() ; + + if ( result.type == T_VARIABLE ) + var2const(); + + switch ( result.type ) + { + case T_CONSTANT: + switch ( type ) + { + case TYPE_FLOAT: + PRIDWORD( inst, 4*i ) = *( int * ) & result.value ; + break; + + case TYPE_INT: + case TYPE_DWORD: + case TYPE_POINTER: + case TYPE_SHORT: + case TYPE_WORD: + case TYPE_BYTE: + case TYPE_SBYTE: + case TYPE_CHAR: + PRIDWORD( inst, 4*i ) = ( int ) result.value ; + break; + + case TYPE_STRING: + default: + instance_destroy( inst ); + console_printf( "¬02Invalid argument %d¬07", i ); + return; + } + break; + + case T_STRING: + PRIDWORD( inst, 4*i ) = ( int ) string_new( result.name ) ; + string_use( PRIDWORD( inst, 4*i ) ); + break; + + case T_VARIABLE: + default: + instance_destroy( inst ); + console_printf( "¬02Invalid argument %d¬07", i ); + return; + } + } + console_printf( "¬07Process %s is executed", p->name ); + } + else + { + console_printf( "¬02Process %s not found¬07", action ); + } + return; + } + } + } + + if ( + strcmp( action, "KILLALL" ) == 0 || + strcmp( action, "SLEEPALL" ) == 0 || + strcmp( action, "WAKEUPALL" ) == 0 || + strcmp( action, "FREEZEALL" ) == 0 ) + { + char act = *action; + int found = 0; + char * oaction = strdup( action ); + char * optr = ptr; + + i = NULL; + while (( i = findproc( i, action, ptr ) ) ) + { + found = 1; + switch ( act ) + { + case 'K': + LOCDWORD( mod_debug, i, STATUS ) = ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) | STATUS_KILLED ; + break; + + case 'W': + LOCDWORD( mod_debug, i, STATUS ) = ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) | STATUS_RUNNING ; + break ; + + case 'S': + LOCDWORD( mod_debug, i, STATUS ) = ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) | STATUS_SLEEPING ; + break ; + + case 'F': + LOCDWORD( mod_debug, i, STATUS ) = ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) | STATUS_FROZEN ; + break; + } + strcpy( action, oaction ); + ptr = optr; + } + + if ( oaction ) free( oaction ); + if ( found ) console_printf( "¬07OK" ); + return ; + } + + if ( + strcmp( action, "KILL" ) == 0 || + strcmp( action, "SLEEP" ) == 0 || + strcmp( action, "WAKEUP" ) == 0 || + strcmp( action, "FREEZE" ) == 0 ) + { + char act = *action; + i = findproc( NULL, action, ptr ); + if ( !i ) return; + + switch ( act ) + { + case 'K': + LOCDWORD( mod_debug, i, STATUS ) = ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) | STATUS_KILLED ; + break; + + case 'W': + LOCDWORD( mod_debug, i, STATUS ) = ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) | STATUS_RUNNING ; + break ; + + case 'S': + LOCDWORD( mod_debug, i, STATUS ) = ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) | STATUS_SLEEPING ; + break ; + + case 'F': + LOCDWORD( mod_debug, i, STATUS ) = ( LOCDWORD( mod_debug, i, STATUS ) & STATUS_WAITING_MASK ) | STATUS_FROZEN ; + break; + } + console_printf( "¬07OK" ); + return ; + } + + if ( strcmp( action, "QUIT" ) == 0 ) + { + bgdrtm_exit( 1 ) ; + return ; + } + + /* Variables del ENGINE */ + + for ( var = 0 ; var < N_CONSOLE_VARS ; var++ ) + { + if ( strcmp( console_vars[var].name, action ) == 0 ) + { + switch ( console_vars[var].type ) + { + case CON_DWORD: + if ( *ptr ) + { + while ( *ptr == '=' || *ptr == ' ' ) ptr++; + eval_expression( ptr, 0 ); + if ( result.type != T_ERROR ) + *( int * )console_vars[var].value = ( int ) result.value ; + } + console_printf( "¬07%s = %d", console_vars[var].name, *( int * )console_vars[var].value ) ; + return ; + + case CON_DWORD_HEX: + if ( *ptr ) + { + while ( *ptr == '=' || *ptr == ' ' ) ptr++; + eval_expression( ptr, 0 ); + if ( result.type != T_ERROR ) + *( int * )console_vars[var].value = ( int ) result.value ; + } + console_printf( "¬07%s = %08Xh\n", console_vars[var].name, *( int * )console_vars[var].value ) ; + return ; + } + } + } + + /* Expresiones */ + + eval_expression( command, 1 ) ; +} + +/* --------------------------------------------------------------------------- */ +/* Hotkey for exit (ALT+X) */ +/* --------------------------------------------------------------------------- */ + +static int force_exit_cb( SDL_keysym k ) +{ + exit_value = 0; + must_exit = 1 ; + return 1; +} + +/* --------------------------------------------------------------------------- */ +/* Hotkeys for activate/deactivate console */ +/* --------------------------------------------------------------------------- */ + +static int console_keyboard_handler_cb( SDL_keysym k ) +{ + char cmd[256]; + + if ( dcb.data.NSourceFiles ) + { + if (( k.mod & KMOD_LALT ) && k.sym == SDLK_c ) + { + if ( !debug_mode ) + { + SDL_EnableKeyRepeat( 250, 50 ); + debug_mode = 1; + force_debug = 1; + console_showing = 1 ; + } + else + { + SDL_EnableKeyRepeat( 0, 0 ); + debug_mode = 0; + force_debug = 0; + console_showing = 0 ; + break_on_next_proc = 0; + } + return 1; + } + + if ( debug_mode ) + { + if ( k.sym == SDLK_F1 ) + { + console_do( "HELP" ); + return 1; + } + + if ( k.mod & KMOD_LALT ) + { + if ( k.sym == SDLK_LEFT ) + { + if ( console_columns > HOTKEYHELP_SIZE ) console_columns-- ; + return 1; + } + + if ( k.sym == SDLK_RIGHT ) + { + if ( console_columns < scrbitmap->width / CHARWIDTH ) console_columns++ ; + return 1; + } + + if ( k.sym == SDLK_UP ) + { + if ( console_lines > 10 ) console_lines-- ; + return 1; + } + + if ( k.sym == SDLK_DOWN ) + { + if ( console_lines < scrbitmap->height / CHARHEIGHT ) console_lines++ ; + return 1; + } + } + + if ( k.mod & ( KMOD_LSHIFT | KMOD_RSHIFT ) ) + { + if ( k.sym == SDLK_LEFT ) + { + if ( console_list_x_pos[console_list_current] > 0 ) console_list_x_pos[console_list_current]-- ; + return 1; + } + + if ( k.sym == SDLK_RIGHT ) + { + console_list_x_pos[console_list_current]++ ; + return 1; + } + + if ( k.sym == SDLK_UP ) + { + if ( console_list_y_pos[console_list_current] > 0 ) console_list_y_pos[console_list_current]-- ; + return 1; + } + + if ( k.sym == SDLK_DOWN ) + { + if ( console_list_y_pos[console_list_current] < console_list_y_max[console_list_current] - 1 ) console_list_y_pos[console_list_current]++ ; + return 1; + } + + if ( k.sym == SDLK_F2 ) + { + console_instance_dump_all_brief(); + return 1; + } + + if ( console_list_current_instance && console_list_current == 1 ) + { + if ( k.sym == SDLK_F3 ) + { + int id = LOCDWORD( mod_debug, console_list_current_instance, PROCESS_ID ) ; + console_printf( "¬04%s (%d) LOCALS¬07\n\n", + ( dcb.data.NSourceFiles && dcb.proc[console_list_current_instance->proc->type].data.ID ) ? getid_name( dcb.proc[console_list_current_instance->proc->type].data.ID ) : (( console_list_current_instance->proc->type == 0 ) ? "Main" : "proc" ), + id + ); + sprintf( cmd, "LOCALS %d", id ) ; + console_do( cmd ); + return 1; + } + + if ( k.sym == SDLK_F4 ) + { + int id = LOCDWORD( mod_debug, console_list_current_instance, PROCESS_ID ) ; + console_printf( "¬04%s (%d) PRIVATES¬07\n\n", + ( dcb.data.NSourceFiles && dcb.proc[console_list_current_instance->proc->type].data.ID ) ? getid_name( dcb.proc[console_list_current_instance->proc->type].data.ID ) : (( console_list_current_instance->proc->type == 0 ) ? "Main" : "proc" ), + id + ); + sprintf( cmd, "PRIVATES %d", id ) ; + console_do( cmd ); + return 1; + } + + if ( k.sym == SDLK_F5 ) + { + int id = LOCDWORD( mod_debug, console_list_current_instance, PROCESS_ID ) ; + console_printf( "¬04%s (%d) PUBLICS¬07\n\n", + ( dcb.data.NSourceFiles && dcb.proc[console_list_current_instance->proc->type].data.ID ) ? getid_name( dcb.proc[console_list_current_instance->proc->type].data.ID ) : (( console_list_current_instance->proc->type == 0 ) ? "Main" : "proc" ), + id + ); + sprintf( cmd, "PUBLICS %d", id ) ; + console_do( cmd ); + return 1; + } + } + + if ( k.sym == SDLK_F6 ) + { + console_list_current ^= 1; + return 1; + } + + if ( k.sym == SDLK_F9 ) + { + switch ( console_list_current ) + { + case 0: + procs[console_list_y_pos[console_list_current]].breakpoint = !procs[console_list_y_pos[console_list_current]].breakpoint; + break; + + case 1: + if ( console_list_current_instance ) console_list_current_instance->breakpoint = !console_list_current_instance->breakpoint; + break; + } + return 1; + } + } + + if ( k.sym == SDLK_F2 ) + { + console_instance_dump_all() ; + return 1; + } + + if ( !( k.mod & ( KMOD_LSHIFT | KMOD_RSHIFT ) ) ) + { + if ( k.sym == SDLK_F5 ) + { + console_do( "GO" ); + return 1; + } + + if ( k.sym == SDLK_F8 ) + { + console_do( "TRACE" ); + return 1; + } + + if ( k.sym == SDLK_F10 ) + { + console_do( "NEXTFRAME" ); + return 1; + } + + if ( k.sym == SDLK_F11 ) + { + console_do( "NEXTPROC" ); + return 1; + } + } + + if ( k.sym == SDLK_PAGEUP ) + { + console_scroll( console_lines ) ; + return 1; + } + + if ( k.sym == SDLK_PAGEDOWN ) + { + console_scroll( -console_lines ) ; + return 1; + } + + if ( k.mod & ( KMOD_RCTRL | KMOD_LCTRL ) ) + { + if ( k.sym == SDLK_LEFT ) + { + console_lateral_scroll( 1 ) ; + return 1; + } + + if ( k.sym == SDLK_RIGHT ) + { + console_lateral_scroll( -1 ) ; + return 1; + } + + if ( k.sym == SDLK_UP ) + { + console_scroll( 1 ) ; + return 1; + } + + if ( k.sym == SDLK_DOWN ) + { + console_scroll( -1 ) ; + return 1; + } + } + + if ( !( k.mod & KMOD_LALT ) ) console_getkey( k.unicode, k.sym ) ; + return 1; + } + } + + return 0; +} + +/* --------------------------------------------------------------------------- */ + +static void console_draw( INSTANCE * i, REGION * clip ) +{ + int x, y, line, count ; + + if ( break_on_next_proc ) return ; + + if ( debug_on_frame || force_debug ) + { + SDL_EnableKeyRepeat( 250, 50 ); + debug_on_frame = 0; + force_debug = 0; + debug_mode = 1; + console_showing = 1 ; + } + + if ( console_columns > scrbitmap->width / CHARWIDTH ) + console_columns = scrbitmap->width / CHARWIDTH ; + + if ( console_lines > ( scrbitmap->height - CHARHEIGHT * 3 ) / CHARHEIGHT ) + console_lines = ( scrbitmap->height - CHARHEIGHT * 3 ) / CHARHEIGHT ; + + if ( console_showing ) + { + if ( console_y < console_lines * CHARHEIGHT ) console_y += CHARHEIGHT ; + if ( console_y > console_lines * CHARHEIGHT ) console_y = console_lines * CHARHEIGHT ; + } + else + { + if ( console_y > 0 ) console_y -= CHARHEIGHT ; + if ( console_y < 0 ) console_y = 0 ; + } + + x = ( scrbitmap->width - console_columns * CHARWIDTH ) / 2 ; + y = -console_lines * CHARHEIGHT + console_y ; + + int current_show = 0; + + for ( count = 0; count < MAX_EXPRESSIONS; count++ ) + { + if ( show_expression[count] ) + { + char * res = eval_expression( show_expression[count], 0 ); + + if ( !res || result.type == T_ERROR ) + { + free( show_expression[count] ); + show_expression[count] = NULL; + show_expression_count--; + } + else + { + int ln = strlen( res ); + int rl = ln * CHARWIDTH; + int xo = ( scrbitmap->width - rl ) / 2; + int yo = ( ( console_y < 1 ) ? 2 : console_y + CHARHEIGHT * 2 + 1 ) + CHARHEIGHT * current_show; + + systext_color( 0, 0 ) ; + systext_puts( scrbitmap, xo - 1, yo - 1, res, ln ); + systext_puts( scrbitmap, xo - 1, yo , res, ln ); + systext_puts( scrbitmap, xo - 1, yo + 1, res, ln ); + systext_puts( scrbitmap, xo , yo + 1, res, ln ); + systext_puts( scrbitmap, xo + 1, yo + 1, res, ln ); + systext_puts( scrbitmap, xo + 1, yo , res, ln ); + systext_puts( scrbitmap, xo + 1, yo - 1, res, ln ); + systext_puts( scrbitmap, xo , yo - 1, res, ln ); + + systext_color( console_showcolor, 0 ) ; + systext_puts( scrbitmap, xo, yo, res, ln ); + + gr_mark_rect( xo, yo, rl, CHARHEIGHT); + current_show++; + } + } + } + + if ( console_y <= 0 ) return ; + + int shiftstatus = GLODWORD( mod_debug, SHIFTSTATUS ) ; + + line = console_tail ; + + for ( count = 0 ; count < console_lines + console_scroll_pos ; count++ ) + { + if ( line == console_head ) break ; + line-- ; + if ( line < 0 ) line = CONSOLE_HISTORY - 1 ; + } + console_scroll_pos = count - console_lines ; + if ( console_scroll_pos < 0 ) console_scroll_pos = 0 ; + + for ( count = 0; count < console_lines; count++ ) + { + systext_color( 0xC0C0C0, 0x000020 ) ; + if ( !console[line] || console_scroll_lateral_pos >= strlen( console[line] ) ) + { + systext_puts( scrbitmap, x, y, "", console_columns ) ; + } + else + { + int pos = 0 ; + int off = 0; + + do + { + if ( console[line][pos] == '¬' ) + { + off += 3; + systext_puts( scrbitmap, x, y, console[line] + pos, 3 ) ; + } + pos++; + } + while ( pos - off < console_scroll_lateral_pos ) ; + + if ( console_scroll_lateral_pos + off < strlen( console[line] ) ) + systext_puts( scrbitmap, x, y, console[line] + console_scroll_lateral_pos + off, console_columns ) ; + else + systext_puts( scrbitmap, x, y, "", console_columns ) ; + } + + y += CHARHEIGHT ; + line++ ; + if ( line == CONSOLE_HISTORY ) line = 0 ; + } + + if ( console_showing && trace_sentence != -1 ) + { + if ( dcb.data.Version < 0x0710 ) + { + if ( trace_instance && instance_exists( trace_instance ) && dcb.sourcecount[trace_sentence >> 24] ) + { + console_printf( "¬07[%s(%d):%d]\n¬14%s¬07\n\n", + trace_instance->proc->name, + LOCDWORD( mod_debug, trace_instance, PROCESS_ID ), + trace_sentence & 0xFFFFFF, + dcb.sourcelines [trace_sentence >> 24] [( trace_sentence & 0xFFFFFF )-1] ) ; + } + } + else + { + if ( trace_instance && instance_exists( trace_instance ) && dcb.sourcecount[trace_sentence >> 20] ) + { + console_printf( "¬07[%s(%d):%d]\n¬14%s¬07\n\n", + trace_instance->proc->name, + LOCDWORD( mod_debug, trace_instance, PROCESS_ID ), + trace_sentence & 0xFFFFF, + dcb.sourcelines [trace_sentence >> 20] [( trace_sentence & 0xFFFFF )-1] ) ; + } + } + debug_on_frame = 0; + force_debug = 0; + debug_next = 0; + trace_sentence = -1; + } + + systext_color( 0xFFFFFF, 0x404040 ) ; + systext_puts( scrbitmap, x, y, ">", 2 ) ; + strcat( console_input, "_" ) ; + systext_puts( scrbitmap, x + CHARWIDTH*2, y, console_input, console_columns - 2 ) ; + console_input[strlen( console_input )-1] = 0 ; + + if ( shiftstatus & 3 ) + { + show_list_window(); + systext_color( 0x404040, 0x00C0C0 ) ; + if ( console_list_current ) + systext_puts( scrbitmap, x, y + CHARHEIGHT, HOTKEYHELP3, console_columns ) ; + else + systext_puts( scrbitmap, x, y + CHARHEIGHT, HOTKEYHELP2, console_columns ) ; + } + else + { + systext_color( 0x404040, 0x00C0C0 ) ; + systext_puts( scrbitmap, x, y + CHARHEIGHT, HOTKEYHELP1, console_columns ) ; + } +} + +/* --------------------------------------------------------------------------- */ + +static int console_info( INSTANCE * i, REGION * clip, int * z, int * drawme ) +{ + * drawme = debug_mode || show_expression_count || ( console_y > 0 ); + + if ( debug_mode || ( console_y > 0 ) ) + { + clip->x = 0; + clip->y = 0; + clip->x2 = scrbitmap->width ; + clip->y2 = scrbitmap->height ; + } + else if ( show_expression_count ) + { + int count, rl = 0, l; + + for ( count = 0; count < MAX_EXPRESSIONS; count++ ) + { + if ( show_expression[count] ) + { + char * res = eval_expression( show_expression[count], 0 ); + + if ( !res ) + { + free( show_expression[count] ); + show_expression[count] = NULL; + show_expression_count--; + } + else + { + if ( ( l = strlen( res ) * CHARWIDTH ) > rl ) rl = l ; + } + } + } + + if ( !show_expression_count ) + { + *drawme = 0; + return 0; + } + + clip->x = ( scrbitmap->width - rl ) / 2 ; + clip->y = ( console_y <= 0 ) ? 0 : ( console_y + CHARHEIGHT ) ; + clip->x2 = clip->x + rl ; + clip->y2 = CHARHEIGHT * show_expression_count; + } + + return * drawme ; +} + +/* --------------------------------------------------------------------------- */ + +static int moddebug_trace( INSTANCE * my, int * params ) +{ + debug = params[0]; + return 0 ; +} + +/* --------------------------------------------------------------------------- */ + +void __bgdexport( mod_debug, process_exec_hook )( INSTANCE * r ) +{ + if ( break_on_next_proc ) + { + debug_next = 1; + break_on_next_proc = 0; + } +} + +/* --------------------------------------------------------------------------- */ +/* Funciones de inicializacion del modulo/plugin */ + +void __bgdexport( mod_debug, module_initialize )() +{ + if ( dcb.data.NSourceFiles ) + { + hotkey_add( KMOD_LALT, SDLK_x, force_exit_cb ); + hotkey_add( 0, 0, console_keyboard_handler_cb ); + + gr_new_object( -2147483647L - 1, console_info, console_draw, 0 ); + } +} + +/* --------------------------------------------------------------------------- */ +/* +void __bgdexport( mod_debug, module_finalize )() +{ +} +*/ +/* --------------------------------------------------------------------------- */ +/* exports */ +/* --------------------------------------------------------------------------- */ + +#include "mod_debug_exports.h" + +/* --------------------------------------------------------------------------- */ |