* Implemented colorized output.

* Implemented finding programs in the PATH.
* Added human readable error codes for status_t and ssize_t return values.


git-svn-id: file:///srv/svn/repos/haiku/trunk/current@11448 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Ingo Weinhold 2005-02-22 12:37:24 +00:00
parent c9b3d41977
commit 592cc30168

View File

@ -62,6 +62,13 @@ static const char *kUsage =
" ID."
;
// terminal color escape sequences
// (http://www.dee.ufcg.edu.br/~rrbrandt/tools/ansi.html)
static const char *kTerminalTextNormal = "\33[0m";
static const char *kTerminalTextRed = "\33[31m";
static const char *kTerminalTextMagenta = "\33[35m";
// command line args
static int sArgc;
static const char *const *sArgv;
@ -108,15 +115,74 @@ get_id(const char *str, int32 &id)
return true;
}
// find_program
static
status_t
find_program(const char *programName, string &resolvedPath)
{
// If the program name is absolute, then there's nothing to do.
// If the program name consists of more than one path element, then we
// consider it a relative path and don't search in PATH either.
if (*programName == '/' || strchr(programName, '/')) {
resolvedPath = programName;
return B_OK;
}
// get the PATH environment variable
const char *paths = getenv("PATH");
if (!paths)
return B_ENTRY_NOT_FOUND;
// iterate through the paths
do {
const char *pathEnd = strchr(paths, ':');
int pathLen = (pathEnd ? pathEnd - paths : strlen(paths));
// We skip empty paths.
if (pathLen > 0) {
// get the program path
string path(paths, pathLen);
path += "/";
path += programName;
// stat() the path to be sure, there is a file
struct stat st;
if (stat(path.c_str(), &st) == 0 && S_ISREG(st.st_mode)) {
resolvedPath = path;
return B_OK;
}
}
paths = (pathEnd ? pathEnd + 1 : NULL);
} while (paths);
// not found in PATH
return B_ENTRY_NOT_FOUND;
}
// load_program
thread_id
load_program(const char *const *args, int32 argCount)
{
// TODO: We need to find the program in the PATH, if no absolute or relative
// path has been given (i.e. only a name).
return load_image(argCount, (const char**)args, (const char**)environ);
}
// clone the argument vector so that we can change it
const char **mutableArgs = new const char*[argCount];
for (int i = 0; i < argCount; i++)
mutableArgs[i] = args[i];
// resolve the program path
string programPath;
status_t error = find_program(args[0], programPath);
if (error != B_OK)
return error;
mutableArgs[0] = programPath.c_str();
// load the program
error = load_image(argCount, mutableArgs, (const char**)environ);
delete[] mutableArgs;
return error;
}
// set_team_debugging_flags
static
@ -258,12 +324,16 @@ print_syscall(debug_post_syscall &message, MemoryReader &memoryReader,
bool printArguments, bool getContents, bool printReturnValue,
bool colorize)
{
// TODO: colorize support
int32 syscallNumber = message.syscall;
Syscall *syscall = sSyscallVector[syscallNumber];
// print syscall name
if (colorize) {
printf("[%6ld] %s%s%s(", message.thread, kTerminalTextRed,
syscall->Name().c_str(), kTerminalTextNormal);
} else {
printf("[%6ld] %s(", message.thread, syscall->Name().c_str());
}
// print arguments
if (printArguments) {
@ -284,15 +354,29 @@ print_syscall(debug_post_syscall &message, MemoryReader &memoryReader,
// print return value
if (printReturnValue) {
TypeHandler *handler = syscall->ReturnType()->Handler();
Type *returnType = syscall->ReturnType();
TypeHandler *handler = returnType->Handler();
string value = handler->GetReturnValue(message.return_value,
getContents, memoryReader);
if (value.length() > 0)
if (value.length() > 0) {
printf(" = %s", value.c_str());
// TODO: If the return value is a status_t, print the error code.
// if the return type is status_t or ssize_t, print human-readable
// error codes
if (returnType->TypeName() == "status_t"
|| returnType->TypeName() == "ssize_t"
&& message.return_value < 0) {
printf(" %s", strerror(message.return_value));
}
}
}
if (colorize) {
printf(" %s(%lld us)%s\n", kTerminalTextMagenta,
message.end_time - message.start_time, kTerminalTextNormal);
} else {
printf(" (%lld us)\n", message.end_time - message.start_time);
}
//for (int32 i = 0; i < 16; i++) {
// if (i % 4 == 0) {
@ -360,6 +444,9 @@ main(int argc, const char *const *argv)
// initialize our syscalls vector and map
init_syscalls();
// don't colorize the output, if we don't have a terminal
colorize = colorize && isatty(STDOUT_FILENO);
// get thread/team to be debugged
thread_id thread = -1;
team_id team = -1;