/* * /+\ * +\ Copyright 1993-2002 Christopher Seiwald and Perforce Software, Inc. * \+/ * * This file is part of jam. * * License is hereby granted to use this software and distribute it * freely, as long as this copyright notice is retained and modifications * are clearly marked. * * ALL WARRANTIES ARE HEREBY DISCLAIMED. */ /* * jam.c - make redux * * See Jam.html for usage information. * * These comments document the code. * * The top half of the code is structured such: * * jam * / | \ * +---+ | \ * / | \ * jamgram option \ * / | \ \ * / | \ \ * / | \ | * scan | compile make * | | / | \ / | \ * | | / | \ / | \ * | | / | \ / | \ * jambase parse | rules search make1 * | | | \ * | | | \ * | | | \ * builtins timestamp command execute * | * | * | * filesys * * * The support routines are called by all of the above, but themselves * are layered thus: * * variable|expand * / | | | * / | | | * / | | | * lists | | pathsys * \ | | * \ | | * \ | | * newstr | * \ | * \ | * \ | * hash * * Roughly, the modules are: * * builtins.c - jam's built-in rules * command.c - maintain lists of commands * compile.c - compile parsed jam statements * execunix.c - execute a shell script on UNIX * execvms.c - execute a shell script, ala VMS * expand.c - expand a buffer, given variable values * file*.c - scan directories and archives on * * hash.c - simple in-memory hashing routines * hcache.c - handle caching of #includes in source files * headers.c - handle #includes in source files * jambase.c - compilable copy of Jambase * jamgram.y - jam grammar * lists.c - maintain lists of strings * make.c - bring a target up to date, once rules are in place * make1.c - execute command to bring targets up to date * newstr.c - string manipulation routines * option.c - command line option processing * parse.c - make and destroy parse trees as driven by the parser * path*.c - manipulate file names on * * hash.c - simple in-memory hashing routines * regexp.c - Henry Spencer's regexp * rules.c - access to RULEs, TARGETs, and ACTIONs * scan.c - the jam yacc scanner * search.c - find a target along $(SEARCH) or $(LOCATE) * timestamp.c - get the timestamp of a file or archive member * variable.c - handle jam multi-element variables * * 05/04/94 (seiwald) - async multiprocess (-j) support * 02/08/95 (seiwald) - -n implies -d2. * 02/22/95 (seiwald) - -v for version info. * 09/11/00 (seiwald) - PATCHLEVEL folded into VERSION. * 01/10/01 (seiwald) - pathsys.h split from filesys.h * 01/21/02 (seiwald) - new -q to quit quickly on build failure * 03/16/02 (seiwald) - support for -g (reorder builds by source time) * 09/19/02 (seiwald) - new -d displays * 10/22/02 (seiwald) - list_new() now does its own newstr()/copystr() * 11/04/02 (seiwald) - const-ing for string literals */ # include "jam.h" # include "option.h" # include "patchlevel.h" /* These get various function declarations. */ # include "lists.h" # include "parse.h" # include "variable.h" # include "compile.h" # include "builtins.h" # include "rules.h" # include "newstr.h" # include "scan.h" # include "timestamp.h" # include "make.h" /* Macintosh is "special" */ # ifdef OS_MAC # include <QuickDraw.h> # endif /* And UNIX for this */ # ifdef unix # include <sys/utsname.h> # endif struct globs globs = { 0, /* noexec */ 1, /* jobs */ 0, /* quitquick */ 0, /* newestfirst */ # ifdef OS_MAC { 0 }, /* display - suppress actions output */ # else { 0, 1 }, /* display actions */ # endif 0, /* output commands, not run them */ 1 /* CWM. Always scanheaders by default */ } ; /* Symbols to be defined as true for use in Jambase */ static const char *othersyms[] = { OSMAJOR, OSMINOR, OSPLAT, JAMVERSYM, 0 } ; /* Known for sure: * mac needs arg_enviro * OS2 needs extern environ */ # ifdef OS_MAC # define use_environ arg_environ # ifdef MPW QDGlobals qd; # endif # endif # ifndef use_environ # define use_environ environ # if !defined( __WATCOM__ ) && !defined( OS_OS2 ) && !defined( OS_NT ) extern char **environ; # endif # endif main( int argc, char **argv, char **arg_environ ) { int n; const char *s; struct option optv[N_OPTS]; const char *all = "all"; int anyhow = 0; int status; #ifdef OPT_PRINT_TOTAL_TIME_EXT time_t start; time(&start); #endif # ifdef OS_MAC InitGraf(&qd.thePort); # endif argc--, argv++; #ifndef OPT_HEADER_OPTION_EXT if( ( n = getoptions( argc, argv, "d:j:f:gs:t:ano:qv", optv ) ) < 0 ) #else /* Added -p */ if( ( n = getoptions( argc, argv, "d:j:f:gs:t:ano:qvp", optv ) ) < 0 ) #endif { printf( "\nusage: jam [ options ] targets...\n\n" ); printf( "-a Build all targets, even if they are current.\n" ); printf( "-dx Display (a)actions (c)causes (d)dependencies\n" ); #ifdef OPT_GRAPH_DEBUG_EXT printf( " (m)make tree (x)commands (g)graph\n" ); printf( " (f)fate changes (0-21) debug levels.\n" ); #else printf( " (m)make tree (x)commands (0-9) debug levels.\n" ); #endif printf( "-fx Read x instead of Jambase.\n" ); printf( "-g Build from newest sources first.\n" ); printf( "-jx Run up to x shell commands concurrently.\n" ); printf( "-n Don't actually execute the updating actions.\n" ); printf( "-ox Write the updating actions to file x.\n" ); #ifdef OPT_HEADER_OPTION_EXT printf( "-p Partial build. Ignore header dependencies\n"); #endif printf( "-q Quit quickly as soon as a target fails.\n" ); printf( "-sx=y Set variable x=y, overriding environment.\n" ); printf( "-tx Rebuild x, even if it is up-to-date.\n" ); printf( "-v Print the version of jam and exit.\n\n" ); printf( "\n" ); #ifdef OPT_GRAPH_DEBUG_EXT /* CWM - Added help on debug levels */ printf( "The debug levels are:\n" ); printf( " 1 - show make actions when executed\n" ); printf( " 2 - show text of actions\n" ); printf( " 3 - show progress of make. Show files when bound\n" ); printf( " 4 - show execcmds()'s work\n" ); printf( " 5 - show rule invocations\n" ); printf( " 6 - show header scan\n" ); printf( " 7 - show dir scan\n" ); printf( " 8 - show attempts at binding\n" ); printf( " 9 - show variable setting\n" ); printf( " 10 - show variable fetches\n" ); printf( " 11 - show variable expansions\n" ); printf( " 12 - show 'if' calculations\n" ); printf( " 13 - show list manipulation\n" ); printf( " 14 - show scanner tokens\n" ); printf( " 15 - show memory use\n" ); printf( " 16 - (-da) show even quiet actions\n" ); printf( " 17 - (-dx) show text of actions\n" ); printf( " 18 - (-dd) show dependency graph\n" ); printf( " 19 - (-dc) show causes of build\n" ); printf( " 20 - (-dg) cwm show dependency graph\n" ); printf( " 21 - (-df) cwm show fate changes\n" ); #endif printf( "\n" ); printf( " To select individual debug levels, try: -d+n -d+n...\n" ); exit( EXITBAD ); } argc -= n, argv += n; /* Version info. */ if( ( s = getoptval( optv, 'v', 0 ) ) ) { printf( "Jam %s. %s. ", VERSION, OSMINOR ); printf( "Copyright 1993-2002 Christopher Seiwald.\n" ); return EXITOK; } /* Pick up interesting options */ if( ( s = getoptval( optv, 'n', 0 ) ) ) globs.noexec++, DEBUG_MAKE = DEBUG_MAKEQ = DEBUG_EXEC = 1; if( ( s = getoptval( optv, 'q', 0 ) ) ) globs.quitquick = 1; if( ( s = getoptval( optv, 'a', 0 ) ) ) anyhow++; #ifdef OPT_HEADER_OPTION_EXT if( ( s = getoptval( optv, 'p', 0 ) ) ) globs.scanheaders = 0; #endif if( ( s = getoptval( optv, 'j', 0 ) ) ) globs.jobs = atoi( s ); if( ( s = getoptval( optv, 'g', 0 ) ) ) globs.newestfirst = 1; /* Turn on/off debugging */ for( n = 0; s = getoptval( optv, 'd', n ); n++ ) { int i = atoi( s ); /* First -d, turn off defaults. */ if( !n ) DEBUG_MAKE = DEBUG_MAKEQ = DEBUG_EXEC = 0; /* n turns on levels 1-n */ /* +n turns on level n */ /* c turns on named display c */ if( i < 0 || i >= DEBUG_MAX ) { printf( "Invalid debug level '%s'.\n", s ); } else if( *s == '+' ) { globs.debug[i] = 1; } else if( i ) while( i ) { globs.debug[i--] = 1; } else while( *s ) switch( *s++ ) { case 'a': DEBUG_MAKE = DEBUG_MAKEQ = 1; break; case 'c': DEBUG_CAUSES = 1; break; case 'd': DEBUG_DEPENDS = 1; break; #ifdef OPT_GRAPH_DEBUG_EXT case 'f': DEBUG_FATE = 1; break; case 'g': DEBUG_GRAPH = 1; break; #endif case 'm': DEBUG_MAKEPROG = 1; break; case 'x': DEBUG_EXEC = 1; break; case '0': break; default: printf( "Invalid debug flag '%c'.\n", s[-1] ); } } /* CWM - support -d2 as an alias for -dx */ if( globs.debug[ 2 ] ) DEBUG_EXEC = 1; /* Set JAMDATE first */ { char buf[ 128 ]; time_t clock; time( &clock ); strcpy( buf, ctime( &clock ) ); /* Trim newline from date */ if( strlen( buf ) == 25 ) buf[ 24 ] = 0; var_set( "JAMDATE", list_new( L0, buf, 0 ), VAR_SET ); #ifdef OPT_TIME_STAMP_OUTPUT_EXT printf( "...starting...%s", ctime ( &clock ) ); fflush( stdout ); #endif } /* And JAMUNAME */ # ifdef unix { struct utsname u; if( uname( &u ) >= 0 ) { LIST *l = L0; l = list_new( l, u.machine, 0 ); l = list_new( l, u.version, 0 ); l = list_new( l, u.release, 0 ); l = list_new( l, u.nodename, 0 ); l = list_new( l, u.sysname, 0 ); var_set( "JAMUNAME", l, VAR_SET ); } } # endif /* unix */ /* * Jam defined variables OS, OSPLAT */ var_defines( othersyms ); /* load up environment variables */ var_defines( (const char **)use_environ ); /* Load up variables set on command line. */ for( n = 0; s = getoptval( optv, 's', n ); n++ ) { const char *symv[2]; symv[0] = s; symv[1] = 0; var_defines( symv ); } /* Initialize built-in rules */ load_builtins(); #ifdef OPT_JOB_SLOT_EXT /* Silently enforce maximum to be two digits */ /* Test against 99 rather than MAXJOBS, as the intent of */ /* this is to protect the fixed size buffers. MAXJOBS is */ /* enforced elsewhere. */ if( globs.jobs > 99 ) globs.jobs = 99; #endif #ifdef OPT_EXPORT_JOBNUM_EXT { LIST *l = 0; int i; char num[3]; /* The number could be negative if the user is trying to be nasty */ if( globs.jobs < 1 ) globs.jobs = 1; /* How many jobs are we allowed to spawn? */ sprintf( num, "%d", globs.jobs ); var_set( "JAM_NUM_JOBS", list_new( 0, newstr( num ), 0 ), VAR_SET ); /* An easy test to see if there are multiple jobs */ if( globs.jobs > 1 ) var_set( "JAM_MULTI_JOBS", list_new( 0, newstr( num ), 0 ), VAR_SET ); /* Create a list of the job nums. ie, -j4 -> 0 1 2 3 */ for( i = 0; i < globs.jobs; ++i ) { sprintf( num, "%d", i ); l = list_new( l, newstr( num ), 0 ); } var_set( "JAM_JOB_LIST", l, VAR_SET ); } #endif /* Parse ruleset */ for( n = 0; s = getoptval( optv, 'f', n ); n++ ) parse_file( s ); if( !n ) parse_file( "+" ); status = yyanyerrors(); /* Manually touch -t targets */ for( n = 0; s = getoptval( optv, 't', n ); n++ ) touchtarget( s ); /* If an output file is specified, set globs.cmdout to that */ if( s = getoptval( optv, 'o', 0 ) ) { if( !( globs.cmdout = fopen( s, "w" ) ) ) { printf( "Failed to write to '%s'\n", s ); exit( EXITBAD ); } globs.noexec++; } /* Now make target */ if( !argc ) status |= make( 1, &all, anyhow ); else status |= make( argc, (const char **)argv, anyhow ); /* Widely scattered cleanup */ var_done(); donerules(); donestamps(); donestr(); /* close cmdout */ if( globs.cmdout ) fclose( globs.cmdout ); #ifdef OPT_TIME_STAMP_OUTPUT_EXT { time_t clock; time( &clock ); printf( "...finished...%s", ctime ( &clock ) ); } #endif #ifdef OPT_PRINT_TOTAL_TIME_EXT { time_t now; int elapsed; time( &now ); elapsed = (int)difftime(now, start); if (elapsed > 10) { int hours = elapsed / 3600; int minutes = (elapsed % 3600) / 60; int seconds = elapsed % 60; printf("...finished in "); if (hours > 0) printf("%d hr ", hours); if (minutes > 0) printf("%d min ", minutes); if (seconds > 0) printf("%d sec ", seconds); printf("\n"); } } #endif return status ? EXITBAD : EXITOK; }
# | Change | User | Description | Committed | |
---|---|---|---|---|---|
#11 | 4110 | Craig Mcpheeters | Some of my changes from work. | ||
#10 | 3183 | Craig Mcpheeters | Integration of my jam mainline branch into my work branch. | ||
#9 | 3182 | Craig Mcpheeters | Update of my work branch. | ||
#8 | 1606 | Craig Mcpheeters |
Integration from my modified jam mainline. This is all the mainline changes into my work branch. |
||
#7 | 1605 | Craig Mcpheeters |
Updated with respect to my copy at work. A few minor updates/enhancements |
||
#6 | 1477 | Craig Mcpheeters |
Integration from my main jam branch. This is 2.4-dev with my extensions. |
||
#5 | 1191 | Craig Mcpheeters | Integration from my public branch | ||
#4 | 1188 | Craig Mcpheeters |
The latest updates from my A|W work copy. Added a -d11 debug mode, which outputs information on fate changes Create jam variables to show job (-jn) information, such as the number of jobs requested, whether or not there is more than one job, and a list of the job slot numbers. These can be used in a variety of ways in supporting multi-job builds Added jobs slot expansion to the action blocks, the sequence !! is replaced by the job slot the action is running in. Modified the special shell handling to include unix. It used to be that on Unix, you could create an action block which exceeded the size that execvp() could handle (perhaps only when the DynamicCommandSize option is enabled.) On NT, all non-default shells are invoked through a intermediate file. This modification does the same for unix shells. Improved the -d+10 debug level, the dependency graph. It now shows the flags on the node as well (NOUPDATE, NOTFILE, etc.) Only issue the '...skipped' messages for nodes which would have been built |
||
#3 | 785 | Craig Mcpheeters |
Integration from //guest/craig_mcpheeters/jam/src/... This work area now contains both the Alias|Wavefront changes, and the latest integrations from the mainline. There are some changes in this area which shouldn't be merged back into the mainline. As I merge this branch back into my jam/src/... branch, I'll leave out a few of the changes. |
||
#2 | 782 | Craig Mcpheeters |
Initial return of the Alias|Wavefront mods to jam 2.2. I made a stab at a configuration system - see the file Jamfile.config. Most of the mods are now enclosed within #ifdef blocks, which kind of pollutes the code, but may make it easier to accept or reject some of these changes. Some of these #ifdefs could disappear completely if they are accepted into the mainline This return implements the following extensions: * header cache * dynamic command block size, allowing for *large* commands * slightly improved warnings and errors * serial output from jam - nice when working with multiple jobs * an extension to variable modifiers: $(>:/) and $(>:\\) * ability to ignore header dependencies for a build (jam -p) * new debug level, -d+10, which outputs the dependency graph * added no-care support to internal nodes. if they fail, dependents are built anyway * time stamps on output * a few minor output modifications * a fix for nt batch file names conflicing when more than one jam runs at a time Each of the above can be enabled/disabled on the command line. For example, to turn on the HeaderCache code: jam -sHeaderCache=1 that is, build jam first, then use that jam to build a new one with the options you want. Some of these changes may have been made in the mainline already, my next step will be to integrate the mainline changes into these ones This return isn't yet ready for prime-time |
||
#1 | 777 | Craig Mcpheeters |
Branch of my jam/src/... area. This new area establishes the integration base for the copy of Jam which has all the changes. |
||
//guest/craig_mcpheeters/jam/src/jam.c | |||||
#2 | 617 | Craig Mcpheeters | Integration from mainline as of @3 | ||
#1 | 616 | Craig Mcpheeters | Integration from Jam mainline, as of @2 |