|
|
/*
see copyright notice in sqrdbg.h */ #include "tier0/basetypes.h"
#include <squirrel.h>
#if !defined( _X360 ) && !defined( _PS3 ) && !defined( POSIX )
#include <winsock.h>
#endif
#include "sqrdbg.h"
#include "sqdbgserver.h"
SQInteger debug_hook(HSQUIRRELVM v); SQInteger error_handler(HSQUIRRELVM v);
#include "serialize_state.inl"
#if defined(VSCRIPT_DLL_EXPORT) || defined(VSQUIRREL_TEST)
#include "memdbgon.h"
#endif
HSQREMOTEDBG sq_rdbg_init(HSQUIRRELVM v,unsigned short port,SQBool autoupdate) { #ifndef _GAMECONSOLE
sockaddr_in bindaddr; #ifdef _WIN32
WSADATA wsadata; if (WSAStartup (MAKEWORD(1,1), &wsadata) != 0){ return NULL; } #endif
SQDbgServer *rdbg = new SQDbgServer(v); rdbg->_autoupdate = autoupdate?true:false; rdbg->_accept = socket(AF_INET,SOCK_STREAM,0); bindaddr.sin_family = AF_INET; bindaddr.sin_port = htons(port); bindaddr.sin_addr.s_addr = htonl (INADDR_ANY); if(bind(rdbg->_accept,(sockaddr*)&bindaddr,sizeof(bindaddr))==SOCKET_ERROR){ delete rdbg; sq_throwerror(v,_SC("failed to bind the socket")); return NULL; } if(!rdbg->Init()) { delete rdbg; sq_throwerror(v,_SC("failed to initialize the debugger")); return NULL; } return rdbg; #else
return NULL; #endif
}
SQRESULT sq_rdbg_waitforconnections(HSQREMOTEDBG rdbg) { #ifndef _GAMECONSOLE
if(SQ_FAILED(sq_compilebuffer(rdbg->_v,serialize_state_nut,(SQInteger)scstrlen(serialize_state_nut),_SC("SERIALIZE_STATE"),SQFalse))) { sq_throwerror(rdbg->_v,_SC("error compiling the serialization function")); } sq_getstackobj(rdbg->_v,-1,&rdbg->_serializefunc); sq_addref(rdbg->_v,&rdbg->_serializefunc); sq_pop(rdbg->_v,1);
sockaddr_in cliaddr; int addrlen=sizeof(cliaddr); if(listen(rdbg->_accept,0)==SOCKET_ERROR) return sq_throwerror(rdbg->_v,_SC("error on listen(socket)")); rdbg->_endpoint = accept(rdbg->_accept,(sockaddr*)&cliaddr,(socklen_t *)&addrlen); //do not accept any other connection
sqdbg_closesocket(rdbg->_accept); rdbg->_accept = INVALID_SOCKET; if(rdbg->_endpoint==INVALID_SOCKET){ return sq_throwerror(rdbg->_v,_SC("error accept(socket)")); } while(!rdbg->_ready){ sq_rdbg_update(rdbg); } #endif
return SQ_OK; }
SQRESULT sq_rdbg_update(HSQREMOTEDBG rdbg) { #ifndef _GAMECONSOLE
#ifdef _WIN32
TIMEVAL time; #else
struct timeval time; #endif
time.tv_sec=0; time.tv_usec=0; fd_set read_flags; FD_ZERO(&read_flags); FD_SET(rdbg->_endpoint, &read_flags); select(NULL/*ignored*/, &read_flags, NULL, NULL, &time);
if(FD_ISSET(rdbg->_endpoint,&read_flags)){ char temp[1024]; int size=0; char c,prev=NULL; memset(&temp,0,sizeof(temp)); int res; FD_CLR(rdbg->_endpoint, &read_flags); while((res = recv(rdbg->_endpoint,&c,1,0))>0){ if(c=='\n')break; if(c!='\r'){ temp[size]=c; prev=c; size++; } } switch(res){ case 0: return sq_throwerror(rdbg->_v,_SC("disconnected")); case SOCKET_ERROR: return sq_throwerror(rdbg->_v,_SC("socket error")); } temp[size]=NULL; temp[size+1]=NULL; rdbg->ParseMsg(temp); } #endif
return SQ_OK; }
SQInteger debug_hook(HSQUIRRELVM v) { SQUserPointer up; intp event_type,line; const SQChar *src,*func; sq_getinteger(v,2,&event_type); sq_getstring(v,3,&src); sq_getinteger(v,4,&line); sq_getstring(v,5,&func); sq_getuserpointer(v,-1,&up); HSQREMOTEDBG rdbg = (HSQREMOTEDBG)up; rdbg->Hook(event_type,line,src,func); if(rdbg->_autoupdate) { if(SQ_FAILED(sq_rdbg_update(rdbg))) return sq_throwerror(v,_SC("socket failed")); } return 0; }
SQInteger error_handler(HSQUIRRELVM v) { SQUserPointer up; const SQChar *sErr=NULL; const SQChar *fn=_SC("unknown"); const SQChar *src=_SC("unknown"); int line=-1; SQStackInfos si; sq_getuserpointer(v,-1,&up); HSQREMOTEDBG rdbg=(HSQREMOTEDBG)up; if(SQ_SUCCEEDED(sq_stackinfos(v,1,&si))) { if(si.funcname)fn=si.funcname; if(si.source)src=si.source; line=si.line; scprintf(_SC("*FUNCTION [%s] %s line [%d]\n"),fn,src,si.line); } if(sq_gettop(v)>=1){ if(SQ_SUCCEEDED(sq_getstring(v,2,&sErr))) { scprintf(_SC("\nAN ERROR HAS OCCURED [%s]\n"),sErr); rdbg->Break(si.line,src,_SC("error"),sErr); } else{ scprintf(_SC("\nAN ERROR HAS OCCURED [unknown]\n")); rdbg->Break(si.line,src,_SC("error"),_SC("unknown")); } } rdbg->BreakExecution(); return 0; }
SQRESULT sq_rdbg_shutdown(HSQREMOTEDBG rdbg) { #ifndef _GAMECONSOLE
delete rdbg; #ifdef _WIN32
WSACleanup(); #endif
#endif
return SQ_OK; }
|