2005-07-29 17:01:00 +02:00
|
|
|
/**CFile****************************************************************
|
|
|
|
|
|
|
|
|
|
FileName [cmdUtils.c]
|
|
|
|
|
|
|
|
|
|
SystemName [ABC: Logic synthesis and verification system.]
|
|
|
|
|
|
|
|
|
|
PackageName [Command processing package.]
|
|
|
|
|
|
|
|
|
|
Synopsis [Various utilities of the command package.]
|
|
|
|
|
|
|
|
|
|
Author [Alan Mishchenko]
|
|
|
|
|
|
|
|
|
|
Affiliation [UC Berkeley]
|
|
|
|
|
|
|
|
|
|
Date [Ver. 1.0. Started - June 20, 2005.]
|
|
|
|
|
|
|
|
|
|
Revision [$Id: cmdUtils.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "mainInt.h"
|
|
|
|
|
#include "abc.h"
|
|
|
|
|
#include "cmdInt.h"
|
|
|
|
|
#include <ctype.h> // proper declaration of isspace
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
/// DECLARATIONS ///
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
static int CmdCommandPrintCompare( Abc_Command ** ppC1, Abc_Command ** ppC2 );
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
2008-01-31 05:01:00 +01:00
|
|
|
/// FUNCTION DEFINITIONS ///
|
2005-07-29 17:01:00 +02:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis []
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
int cmdCheckShellEscape( Abc_Frame_t * pAbc, int argc, char ** argv)
|
|
|
|
|
{
|
|
|
|
|
if (argv[0][0] == '!')
|
|
|
|
|
{
|
|
|
|
|
const int size = 4096;
|
|
|
|
|
int i;
|
|
|
|
|
char buffer[4096];
|
|
|
|
|
strncpy (buffer, &argv[0][1], size);
|
|
|
|
|
for (i = 1; i < argc; ++i)
|
|
|
|
|
{
|
|
|
|
|
strncat (buffer, " ", size);
|
|
|
|
|
strncat (buffer, argv[i], size);
|
|
|
|
|
}
|
|
|
|
|
if (buffer[0] == 0)
|
|
|
|
|
strncpy (buffer, "/bin/sh", size);
|
|
|
|
|
system (buffer);
|
|
|
|
|
|
|
|
|
|
// NOTE: Since we reconstruct the cmdline by concatenating
|
|
|
|
|
// the parts, we lose information. So a command like
|
|
|
|
|
// `!ls "file name"` will be sent to the system as
|
|
|
|
|
// `ls file name` which is a BUG
|
|
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Executes one command.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
int CmdCommandDispatch( Abc_Frame_t * pAbc, int argc, char **argv )
|
|
|
|
|
{
|
|
|
|
|
Abc_Ntk_t * pNetCopy;
|
|
|
|
|
int (*pFunc) ( Abc_Frame_t *, int, char ** );
|
|
|
|
|
Abc_Command * pCommand;
|
|
|
|
|
char * value;
|
|
|
|
|
int fError;
|
|
|
|
|
int clk;
|
|
|
|
|
|
|
|
|
|
if ( argc == 0 )
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
if ( cmdCheckShellEscape( pAbc, argc, argv ) == 1 )
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
// get the command
|
|
|
|
|
if ( !st_lookup( pAbc->tCommands, argv[0], (char **)&pCommand ) )
|
|
|
|
|
{ // the command is not in the table
|
|
|
|
|
fprintf( pAbc->Err, "** cmd error: unknown command '%s'\n", argv[0] );
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// get the backup network if the command is going to change the network
|
|
|
|
|
if ( pCommand->fChange )
|
|
|
|
|
{
|
2005-09-05 17:01:00 +02:00
|
|
|
if ( pAbc->pNtkCur && Abc_FrameIsFlagEnabled( "backup" ) )
|
2005-07-29 17:01:00 +02:00
|
|
|
{
|
|
|
|
|
pNetCopy = Abc_NtkDup( pAbc->pNtkCur );
|
|
|
|
|
Abc_FrameSetCurrentNetwork( pAbc, pNetCopy );
|
|
|
|
|
// swap the current network and the backup network
|
|
|
|
|
// to prevent the effect of resetting the short names
|
|
|
|
|
Abc_FrameSwapCurrentAndBackup( pAbc );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// execute the command
|
2008-01-31 05:01:00 +01:00
|
|
|
clk = Extra_CpuTime();
|
2005-09-05 17:01:00 +02:00
|
|
|
pFunc = (int (*)(Abc_Frame_t *, int, char **))pCommand->pFunc;
|
2005-07-29 17:01:00 +02:00
|
|
|
fError = (*pFunc)( pAbc, argc, argv );
|
2008-01-31 05:01:00 +01:00
|
|
|
pAbc->TimeCommand += (Extra_CpuTime() - clk);
|
2005-07-29 17:01:00 +02:00
|
|
|
|
|
|
|
|
// automatic execution of arbitrary command after each command
|
|
|
|
|
// usually this is a passive command ...
|
|
|
|
|
if ( fError == 0 && !pAbc->fAutoexac )
|
|
|
|
|
{
|
|
|
|
|
if ( st_lookup( pAbc->tFlags, "autoexec", &value ) )
|
|
|
|
|
{
|
|
|
|
|
pAbc->fAutoexac = 1;
|
|
|
|
|
fError = Cmd_CommandExecute( pAbc, value );
|
|
|
|
|
pAbc->fAutoexac = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return fError;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Splits the command line string into individual commands.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
char * CmdSplitLine( Abc_Frame_t * pAbc, char *sCommand, int *argc, char ***argv )
|
|
|
|
|
{
|
|
|
|
|
char *p, *start, c;
|
|
|
|
|
int i, j;
|
|
|
|
|
char *new_arg;
|
|
|
|
|
Vec_Ptr_t * vArgs;
|
|
|
|
|
int single_quote, double_quote;
|
|
|
|
|
|
|
|
|
|
vArgs = Vec_PtrAlloc( 10 );
|
|
|
|
|
|
|
|
|
|
p = sCommand;
|
|
|
|
|
for ( ;; )
|
|
|
|
|
{
|
|
|
|
|
// skip leading white space
|
|
|
|
|
while ( isspace( ( int ) *p ) )
|
|
|
|
|
{
|
|
|
|
|
p++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// skip until end of this token
|
|
|
|
|
single_quote = double_quote = 0;
|
|
|
|
|
for ( start = p; ( c = *p ) != '\0'; p++ )
|
|
|
|
|
{
|
|
|
|
|
if ( c == ';' || c == '#' || isspace( ( int ) c ) )
|
|
|
|
|
{
|
|
|
|
|
if ( !single_quote && !double_quote )
|
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ( c == '\'' )
|
|
|
|
|
{
|
|
|
|
|
single_quote = !single_quote;
|
|
|
|
|
}
|
|
|
|
|
if ( c == '"' )
|
|
|
|
|
{
|
|
|
|
|
double_quote = !double_quote;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ( single_quote || double_quote )
|
|
|
|
|
{
|
|
|
|
|
( void ) fprintf( pAbc->Err, "** cmd warning: ignoring unbalanced quote ...\n" );
|
|
|
|
|
}
|
|
|
|
|
if ( start == p )
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
new_arg = ALLOC( char, p - start + 1 );
|
|
|
|
|
j = 0;
|
|
|
|
|
for ( i = 0; i < p - start; i++ )
|
|
|
|
|
{
|
|
|
|
|
c = start[i];
|
|
|
|
|
if ( ( c != '\'' ) && ( c != '\"' ) )
|
|
|
|
|
{
|
|
|
|
|
new_arg[j++] = isspace( ( int ) c ) ? ' ' : start[i];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
new_arg[j] = '\0';
|
|
|
|
|
Vec_PtrPush( vArgs, new_arg );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*argc = vArgs->nSize;
|
|
|
|
|
*argv = (char **)Vec_PtrReleaseArray( vArgs );
|
|
|
|
|
Vec_PtrFree( vArgs );
|
|
|
|
|
if ( *p == ';' )
|
|
|
|
|
{
|
|
|
|
|
p++;
|
|
|
|
|
}
|
|
|
|
|
else if ( *p == '#' )
|
|
|
|
|
{
|
|
|
|
|
for ( ; *p != 0; p++ ); // skip to end of line
|
|
|
|
|
}
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Replaces parts of the command line string by aliases if given.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
int CmdApplyAlias( Abc_Frame_t * pAbc, int *argcp, char ***argvp, int *loop )
|
|
|
|
|
{
|
|
|
|
|
int i, argc, stopit, added, offset, did_subst, subst, fError, newc, j;
|
|
|
|
|
char *arg, **argv, **newv;
|
|
|
|
|
Abc_Alias *alias;
|
|
|
|
|
|
|
|
|
|
argc = *argcp;
|
|
|
|
|
argv = *argvp;
|
|
|
|
|
stopit = 0;
|
2008-01-31 05:01:00 +01:00
|
|
|
for ( ; *loop < 200; ( *loop )++ )
|
2005-07-29 17:01:00 +02:00
|
|
|
{
|
|
|
|
|
if ( argc == 0 )
|
|
|
|
|
return 0;
|
|
|
|
|
if ( stopit != 0 || st_lookup( pAbc->tAliases, argv[0], (char **) &alias ) == 0 )
|
|
|
|
|
{
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
if ( strcmp( argv[0], alias->argv[0] ) == 0 )
|
|
|
|
|
{
|
|
|
|
|
stopit = 1;
|
|
|
|
|
}
|
|
|
|
|
FREE( argv[0] );
|
|
|
|
|
added = alias->argc - 1;
|
|
|
|
|
|
|
|
|
|
/* shift all the arguments to the right */
|
|
|
|
|
if ( added != 0 )
|
|
|
|
|
{
|
|
|
|
|
argv = REALLOC( char *, argv, argc + added );
|
|
|
|
|
for ( i = argc - 1; i >= 1; i-- )
|
|
|
|
|
{
|
|
|
|
|
argv[i + added] = argv[i];
|
|
|
|
|
}
|
|
|
|
|
for ( i = 1; i <= added; i++ )
|
|
|
|
|
{
|
2008-01-31 05:01:00 +01:00
|
|
|
argv[i] = NULL;
|
2005-07-29 17:01:00 +02:00
|
|
|
}
|
|
|
|
|
argc += added;
|
|
|
|
|
}
|
|
|
|
|
subst = 0;
|
|
|
|
|
for ( i = 0, offset = 0; i < alias->argc; i++, offset++ )
|
|
|
|
|
{
|
|
|
|
|
arg = CmdHistorySubstitution( pAbc, alias->argv[i], &did_subst );
|
2008-01-31 05:01:00 +01:00
|
|
|
if ( arg == NULL )
|
2005-07-29 17:01:00 +02:00
|
|
|
{
|
|
|
|
|
*argcp = argc;
|
|
|
|
|
*argvp = argv;
|
|
|
|
|
return ( 1 );
|
|
|
|
|
}
|
|
|
|
|
if ( did_subst != 0 )
|
|
|
|
|
{
|
|
|
|
|
subst = 1;
|
|
|
|
|
}
|
|
|
|
|
fError = 0;
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
arg = CmdSplitLine( pAbc, arg, &newc, &newv );
|
|
|
|
|
/*
|
|
|
|
|
* If there's a complete `;' terminated command in `arg',
|
|
|
|
|
* when split_line() returns arg[0] != '\0'.
|
|
|
|
|
*/
|
|
|
|
|
if ( arg[0] == '\0' )
|
|
|
|
|
{ /* just a bunch of words */
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
fError = CmdApplyAlias( pAbc, &newc, &newv, loop );
|
|
|
|
|
if ( fError == 0 )
|
|
|
|
|
{
|
|
|
|
|
fError = CmdCommandDispatch( pAbc, newc, newv );
|
|
|
|
|
}
|
|
|
|
|
CmdFreeArgv( newc, newv );
|
|
|
|
|
}
|
|
|
|
|
while ( fError == 0 );
|
|
|
|
|
if ( fError != 0 )
|
|
|
|
|
{
|
|
|
|
|
*argcp = argc;
|
|
|
|
|
*argvp = argv;
|
|
|
|
|
return ( 1 );
|
|
|
|
|
}
|
|
|
|
|
added = newc - 1;
|
|
|
|
|
if ( added != 0 )
|
|
|
|
|
{
|
|
|
|
|
argv = REALLOC( char *, argv, argc + added );
|
|
|
|
|
for ( j = argc - 1; j > offset; j-- )
|
|
|
|
|
{
|
|
|
|
|
argv[j + added] = argv[j];
|
|
|
|
|
}
|
|
|
|
|
argc += added;
|
|
|
|
|
}
|
|
|
|
|
for ( j = 0; j <= added; j++ )
|
|
|
|
|
{
|
|
|
|
|
argv[j + offset] = newv[j];
|
|
|
|
|
}
|
|
|
|
|
FREE( newv );
|
|
|
|
|
offset += added;
|
|
|
|
|
}
|
|
|
|
|
if ( subst == 1 )
|
|
|
|
|
{
|
|
|
|
|
for ( i = offset; i < argc; i++ )
|
|
|
|
|
{
|
|
|
|
|
FREE( argv[i] );
|
|
|
|
|
}
|
|
|
|
|
argc = offset;
|
|
|
|
|
}
|
|
|
|
|
*argcp = argc;
|
|
|
|
|
*argvp = argv;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf( pAbc->Err, "** cmd warning: alias loop\n" );
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Performs history substitution (now, disabled).]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
char * CmdHistorySubstitution( Abc_Frame_t * pAbc, char *line, int *changed )
|
|
|
|
|
{
|
|
|
|
|
// as of today, no history substitution
|
|
|
|
|
*changed = 0;
|
|
|
|
|
return line;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Opens the file with path (now, disabled).]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
FILE * CmdFileOpen( Abc_Frame_t * pAbc, char *sFileName, char *sMode, char **pFileNameReal, int silent )
|
|
|
|
|
{
|
|
|
|
|
char * sRealName, * sPathUsr, * sPathLib, * sPathAll;
|
|
|
|
|
FILE * pFile;
|
|
|
|
|
|
|
|
|
|
if (strcmp(sFileName, "-") == 0) {
|
|
|
|
|
if (strcmp(sMode, "w") == 0) {
|
2008-01-31 05:01:00 +01:00
|
|
|
sRealName = Extra_UtilStrsav( "stdout" );
|
2005-07-29 17:01:00 +02:00
|
|
|
pFile = stdout;
|
|
|
|
|
}
|
|
|
|
|
else {
|
2008-01-31 05:01:00 +01:00
|
|
|
sRealName = Extra_UtilStrsav( "stdin" );
|
2005-07-29 17:01:00 +02:00
|
|
|
pFile = stdin;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
sRealName = NULL;
|
|
|
|
|
if (strcmp(sMode, "r") == 0) {
|
|
|
|
|
|
|
|
|
|
/* combine both pathes if exist */
|
|
|
|
|
sPathUsr = Cmd_FlagReadByName(pAbc,"open_path");
|
|
|
|
|
sPathLib = Cmd_FlagReadByName(pAbc,"lib_path");
|
|
|
|
|
|
|
|
|
|
if ( sPathUsr == NULL && sPathLib == NULL ) {
|
|
|
|
|
sPathAll = NULL;
|
|
|
|
|
}
|
|
|
|
|
else if ( sPathUsr == NULL ) {
|
2008-01-31 05:01:00 +01:00
|
|
|
sPathAll = Extra_UtilStrsav( sPathLib );
|
2005-07-29 17:01:00 +02:00
|
|
|
}
|
|
|
|
|
else if ( sPathLib == NULL ) {
|
2008-01-31 05:01:00 +01:00
|
|
|
sPathAll = Extra_UtilStrsav( sPathUsr );
|
2005-07-29 17:01:00 +02:00
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
sPathAll = ALLOC( char, strlen(sPathLib)+strlen(sPathUsr)+5 );
|
|
|
|
|
sprintf( sPathAll, "%s:%s",sPathUsr, sPathLib );
|
|
|
|
|
}
|
2008-01-31 05:01:00 +01:00
|
|
|
if ( sPathAll != NULL ) {
|
|
|
|
|
sRealName = Extra_UtilFileSearch(sFileName, sPathAll, "r");
|
2005-07-29 17:01:00 +02:00
|
|
|
FREE( sPathAll );
|
|
|
|
|
}
|
|
|
|
|
}
|
2008-01-31 05:01:00 +01:00
|
|
|
if (sRealName == NULL) {
|
|
|
|
|
sRealName = Extra_UtilTildeExpand(sFileName);
|
2005-07-29 17:01:00 +02:00
|
|
|
}
|
2008-01-31 05:01:00 +01:00
|
|
|
if ((pFile = fopen(sRealName, sMode)) == NULL) {
|
2005-07-29 17:01:00 +02:00
|
|
|
if (! silent) {
|
|
|
|
|
perror(sRealName);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ( pFileNameReal )
|
|
|
|
|
*pFileNameReal = sRealName;
|
|
|
|
|
else
|
|
|
|
|
FREE(sRealName);
|
|
|
|
|
|
|
|
|
|
return pFile;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Frees the previously allocated argv array.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
void CmdFreeArgv( int argc, char **argv )
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
for ( i = 0; i < argc; i++ )
|
|
|
|
|
FREE( argv[i] );
|
|
|
|
|
FREE( argv );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Frees the previously allocated command.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
void CmdCommandFree( Abc_Command * pCommand )
|
|
|
|
|
{
|
|
|
|
|
free( pCommand->sGroup );
|
|
|
|
|
free( pCommand->sName );
|
|
|
|
|
free( pCommand );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Prints commands alphabetically by group.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
void CmdCommandPrint( Abc_Frame_t * pAbc, bool fPrintAll )
|
|
|
|
|
{
|
|
|
|
|
char *key, *value;
|
|
|
|
|
st_generator * gen;
|
|
|
|
|
Abc_Command ** ppCommands;
|
|
|
|
|
Abc_Command * pCommands;
|
|
|
|
|
int nCommands, i;
|
|
|
|
|
char * sGroupCur;
|
|
|
|
|
int LenghtMax, nColumns, iCom = 0;
|
|
|
|
|
|
|
|
|
|
// put all commands into one array
|
|
|
|
|
nCommands = st_count( pAbc->tCommands );
|
|
|
|
|
ppCommands = ALLOC( Abc_Command *, nCommands );
|
|
|
|
|
i = 0;
|
|
|
|
|
st_foreach_item( pAbc->tCommands, gen, &key, &value )
|
|
|
|
|
{
|
|
|
|
|
pCommands = (Abc_Command *)value;
|
|
|
|
|
if ( fPrintAll || pCommands->sName[0] != '_' )
|
|
|
|
|
ppCommands[i++] = pCommands;
|
|
|
|
|
}
|
|
|
|
|
nCommands = i;
|
|
|
|
|
|
|
|
|
|
// sort command by group and then by name, alphabetically
|
|
|
|
|
qsort( (void *)ppCommands, nCommands, sizeof(Abc_Command *),
|
|
|
|
|
(int (*)(const void *, const void *)) CmdCommandPrintCompare );
|
|
|
|
|
assert( CmdCommandPrintCompare( ppCommands, ppCommands + nCommands - 1 ) <= 0 );
|
|
|
|
|
|
|
|
|
|
// get the longest command name
|
|
|
|
|
LenghtMax = 0;
|
|
|
|
|
for ( i = 0; i < nCommands; i++ )
|
|
|
|
|
if ( LenghtMax < (int)strlen(ppCommands[i]->sName) )
|
|
|
|
|
LenghtMax = (int)strlen(ppCommands[i]->sName);
|
|
|
|
|
// get the number of columns
|
|
|
|
|
nColumns = 79 / (LenghtMax + 2);
|
|
|
|
|
|
|
|
|
|
// print the starting message
|
|
|
|
|
fprintf( pAbc->Out, " Welcome to ABC!" );
|
|
|
|
|
|
|
|
|
|
// print the command by group
|
|
|
|
|
sGroupCur = NULL;
|
|
|
|
|
for ( i = 0; i < nCommands; i++ )
|
|
|
|
|
if ( sGroupCur && strcmp( sGroupCur, ppCommands[i]->sGroup ) == 0 )
|
|
|
|
|
{ // this command belongs to the same group as the previous one
|
|
|
|
|
if ( iCom++ % nColumns == 0 )
|
|
|
|
|
fprintf( pAbc->Out, "\n" );
|
|
|
|
|
// print this command
|
|
|
|
|
fprintf( pAbc->Out, " %-*s", LenghtMax, ppCommands[i]->sName );
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{ // this command starts the new group of commands
|
|
|
|
|
// start the new group
|
|
|
|
|
fprintf( pAbc->Out, "\n" );
|
|
|
|
|
fprintf( pAbc->Out, "\n" );
|
|
|
|
|
fprintf( pAbc->Out, "%s commands:\n", ppCommands[i]->sGroup );
|
|
|
|
|
// print this command
|
|
|
|
|
fprintf( pAbc->Out, " %-*s", LenghtMax, ppCommands[i]->sName );
|
|
|
|
|
// remember current command group
|
|
|
|
|
sGroupCur = ppCommands[i]->sGroup;
|
|
|
|
|
// reset the command counter
|
|
|
|
|
iCom = 1;
|
|
|
|
|
}
|
|
|
|
|
fprintf( pAbc->Out, "\n" );
|
|
|
|
|
FREE( ppCommands );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Comparision function used for sorting commands.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
int CmdCommandPrintCompare( Abc_Command ** ppC1, Abc_Command ** ppC2 )
|
|
|
|
|
{
|
|
|
|
|
Abc_Command * pC1 = *ppC1;
|
|
|
|
|
Abc_Command * pC2 = *ppC2;
|
|
|
|
|
int RetValue;
|
|
|
|
|
|
|
|
|
|
RetValue = strcmp( pC1->sGroup, pC2->sGroup );
|
|
|
|
|
if ( RetValue < 0 )
|
|
|
|
|
return -1;
|
|
|
|
|
if ( RetValue > 0 )
|
|
|
|
|
return 1;
|
|
|
|
|
// the command belong to the same group
|
|
|
|
|
|
|
|
|
|
// put commands with "_" at the end of the list
|
|
|
|
|
if ( pC1->sName[0] != '_' && pC2->sName[0] == '_' )
|
|
|
|
|
return -1;
|
|
|
|
|
if ( pC1->sName[0] == '_' && pC2->sName[0] != '_' )
|
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
|
|
RetValue = strcmp( pC1->sName, pC2->sName );
|
|
|
|
|
if ( RetValue < 0 )
|
|
|
|
|
return -1;
|
|
|
|
|
if ( RetValue > 0 )
|
|
|
|
|
return 1;
|
|
|
|
|
// should not be two indentical commands
|
|
|
|
|
assert( 0 );
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2005-09-05 17:01:00 +02:00
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Comparision function used for sorting commands.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
int CmdNamePrintCompare( char ** ppC1, char ** ppC2 )
|
|
|
|
|
{
|
|
|
|
|
return strcmp( *ppC1, *ppC2 );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
|
|
|
|
|
|
Synopsis [Comparision function used for sorting commands.]
|
|
|
|
|
|
|
|
|
|
Description []
|
|
|
|
|
|
|
|
|
|
SideEffects []
|
|
|
|
|
|
|
|
|
|
SeeAlso []
|
|
|
|
|
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
void CmdPrintTable( st_table * tTable, int fAliases )
|
|
|
|
|
{
|
|
|
|
|
st_generator * gen;
|
|
|
|
|
char ** ppNames;
|
|
|
|
|
char * key, * value;
|
|
|
|
|
int nNames, i;
|
|
|
|
|
|
|
|
|
|
// collect keys in the array
|
|
|
|
|
ppNames = ALLOC( char *, st_count(tTable) );
|
|
|
|
|
nNames = 0;
|
|
|
|
|
st_foreach_item( tTable, gen, &key, &value )
|
|
|
|
|
ppNames[nNames++] = key;
|
|
|
|
|
|
|
|
|
|
// sort array by name
|
|
|
|
|
qsort( (void *)ppNames, nNames, sizeof(char *),
|
|
|
|
|
(int (*)(const void *, const void *))CmdNamePrintCompare );
|
|
|
|
|
|
|
|
|
|
// print in this order
|
|
|
|
|
for ( i = 0; i < nNames; i++ )
|
|
|
|
|
{
|
|
|
|
|
st_lookup( tTable, ppNames[i], &value );
|
|
|
|
|
if ( fAliases )
|
|
|
|
|
CmdCommandAliasPrint( Abc_FrameGetGlobalFrame(), (Abc_Alias *)value );
|
|
|
|
|
else
|
|
|
|
|
fprintf( stdout, "%-15s %-15s\n", ppNames[i], value );
|
|
|
|
|
}
|
|
|
|
|
free( ppNames );
|
|
|
|
|
}
|
2005-07-29 17:01:00 +02:00
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
/// END OF FILE ///
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|