From 69822f0c5b67161b4d7558081bc94f6f3a7c5dbb Mon Sep 17 00:00:00 2001 From: Ben Gamari Date: Mon, 16 Nov 2015 16:58:41 +0100 Subject: RtsFlags: Refactor some of the deeper switches This was previously nearly impossible to read; now it's merely difficult. Ideally we would do a more thorough refactoring of the RTS command line parser (#4243) but this is more effort than I have time for at the moment. Test Plan: Try using affected RTS flags Reviewers: simonmar, austin Subscribers: thomie Differential Revision: https://phabricator.haskell.org/D1476 GHC Trac Issues: #4243 --- rts/RtsFlags.c | 435 +++++++++++++++++++++++++++++++-------------------------- 1 file changed, 236 insertions(+), 199 deletions(-) (limited to 'rts/RtsFlags.c') diff --git a/rts/RtsFlags.c b/rts/RtsFlags.c index c349ec3008..b9979bcf43 100644 --- a/rts/RtsFlags.c +++ b/rts/RtsFlags.c @@ -95,8 +95,20 @@ static StgWord64 decodeSize ( static void bad_option (const char *s); +#ifdef DEBUG +static void read_debug_flags(const char *arg); +#endif + +#ifdef PROFILING +static rtsBool read_heap_profiling_flag(const char *arg); +#endif + +#ifdef USE_PAPI +static void read_papi_flag(const char *arg); +#endif + #ifdef TRACING -static void read_trace_flags(char *arg); +static void read_trace_flags(const char *arg); #endif static void errorUsage (void) GNU_ATTRIBUTE(__noreturn__); @@ -771,7 +783,7 @@ error = rtsTrue; = decodeSize(rts_argv[arg], 2, 2*BLOCK_SIZE, HS_INT_MAX) / BLOCK_SIZE; break; - case 'n': + case 'n': OPTION_UNSAFE; RtsFlags.GcFlags.nurseryChunkSize = decodeSize(rts_argv[arg], 2, 2*BLOCK_SIZE, HS_INT_MAX) @@ -781,39 +793,7 @@ error = rtsTrue; #ifdef USE_PAPI case 'a': OPTION_UNSAFE; - switch(rts_argv[arg][2]) { - case '1': - RtsFlags.PapiFlags.eventType = PAPI_FLAG_CACHE_L1; - break; - case '2': - RtsFlags.PapiFlags.eventType = PAPI_FLAG_CACHE_L2; - break; - case 'b': - RtsFlags.PapiFlags.eventType = PAPI_FLAG_BRANCH; - break; - case 's': - RtsFlags.PapiFlags.eventType = PAPI_FLAG_STALLS; - break; - case 'e': - RtsFlags.PapiFlags.eventType = PAPI_FLAG_CB_EVENTS; - break; - case '+': - case '#': - if (RtsFlags.PapiFlags.numUserEvents >= MAX_PAPI_USER_EVENTS) { - errorBelch("maximum number of PAPI events reached"); - stg_exit(EXIT_FAILURE); - } - nat eventNum = RtsFlags.PapiFlags.numUserEvents++; - char kind = rts_argv[arg][2]; - nat eventKind = kind == '+' ? PAPI_PRESET_EVENT_KIND : PAPI_NATIVE_EVENT_KIND; - - RtsFlags.PapiFlags.userEvents[eventNum] = rts_argv[arg] + 3; - RtsFlags.PapiFlags.eventType = PAPI_USER_EVENTS; - RtsFlags.PapiFlags.userEventsKind[eventNum] = eventKind; - break; - default: - bad_option( rts_argv[arg] ); - } + read_papi_flag(&rts_argv[arg]) break; #endif @@ -849,65 +829,7 @@ error = rtsTrue; case 'D': OPTION_SAFE; - DEBUG_BUILD_ONLY( - { - char *c; - - for (c = rts_argv[arg] + 2; *c != '\0'; c++) { - switch (*c) { - case 's': - RtsFlags.DebugFlags.scheduler = rtsTrue; - break; - case 'i': - RtsFlags.DebugFlags.interpreter = rtsTrue; - break; - case 'w': - RtsFlags.DebugFlags.weak = rtsTrue; - break; - case 'G': - RtsFlags.DebugFlags.gccafs = rtsTrue; - break; - case 'g': - RtsFlags.DebugFlags.gc = rtsTrue; - break; - case 'b': - RtsFlags.DebugFlags.block_alloc = rtsTrue; - break; - case 'S': - RtsFlags.DebugFlags.sanity = rtsTrue; - break; - case 't': - RtsFlags.DebugFlags.stable = rtsTrue; - break; - case 'p': - RtsFlags.DebugFlags.prof = rtsTrue; - break; - case 'l': - RtsFlags.DebugFlags.linker = rtsTrue; - break; - case 'a': - RtsFlags.DebugFlags.apply = rtsTrue; - break; - case 'm': - RtsFlags.DebugFlags.stm = rtsTrue; - break; - case 'z': - RtsFlags.DebugFlags.squeeze = rtsTrue; - break; - case 'c': - RtsFlags.DebugFlags.hpc = rtsTrue; - break; - case 'r': - RtsFlags.DebugFlags.sparks = rtsTrue; - break; - default: - bad_option( rts_argv[arg] ); - } - } - // -Dx also turns on -v. Use -l to direct trace - // events to the .eventlog file instead. - RtsFlags.TraceFlags.tracing = TRACE_STDERR; - }) + DEBUG_BUILD_ONLY(read_debug_flags(rts_argv[arg]);) break; case 'K': @@ -1099,109 +1021,8 @@ error = rtsTrue; #else OPTION_SAFE; PROFILING_BUILD_ONLY( - switch (rts_argv[arg][2]) { - case '\0': - case 'C': - case 'c': - case 'M': - case 'm': - case 'D': - case 'd': - case 'Y': - case 'y': - case 'R': - case 'r': - case 'B': - case 'b': - if (rts_argv[arg][2] != '\0' && rts_argv[arg][3] != '\0') { - { - char *left = strchr(rts_argv[arg], '{'); - char *right = strrchr(rts_argv[arg], '}'); - - // curly braces are optional, for - // backwards compat. - if (left) - left = left+1; - else - left = rts_argv[arg] + 3; - - if (!right) - right = rts_argv[arg] + strlen(rts_argv[arg]); - - *right = '\0'; - - switch (rts_argv[arg][2]) { - case 'c': // cost centre label select - RtsFlags.ProfFlags.ccSelector = left; - break; - case 'C': - RtsFlags.ProfFlags.ccsSelector = left; - break; - case 'M': - case 'm': // cost centre module select - RtsFlags.ProfFlags.modSelector = left; - break; - case 'D': - case 'd': // closure descr select - RtsFlags.ProfFlags.descrSelector = left; - break; - case 'Y': - case 'y': // closure type select - RtsFlags.ProfFlags.typeSelector = left; - break; - case 'R': - case 'r': // retainer select - RtsFlags.ProfFlags.retainerSelector = left; - break; - case 'B': - case 'b': // biography select - RtsFlags.ProfFlags.bioSelector = left; - break; - } - } - break; - } - - if (RtsFlags.ProfFlags.doHeapProfile != 0) { - errorBelch("multiple heap profile options"); - error = rtsTrue; - break; - } - - switch (rts_argv[arg][2]) { - case '\0': - case 'C': - case 'c': - RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_CCS; - break; - case 'M': - case 'm': - RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_MOD; - break; - case 'D': - case 'd': - RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_DESCR; - break; - case 'Y': - case 'y': - RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_TYPE; - break; - case 'R': - case 'r': - RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_RETAINER; - break; - case 'B': - case 'b': - RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_LDV; - break; - } - break; - - default: - errorBelch("invalid heap profile option: %s",rts_argv[arg]); - error = rtsTrue; - } - ) + error = read_heap_profiling_flag(rts_argv[arg]); + ); #endif /* PROFILING */ break; @@ -1676,10 +1497,226 @@ decodeSize(const char *flag, nat offset, StgWord64 min, StgWord64 max) return val; } +#ifdef DEBUG +static void read_debug_flags(const char* arg) +{ + // Already parsed "-D" + const char *c; + for (c = arg + 2; *c != '\0'; c++) { + switch (*c) { + case 's': + RtsFlags.DebugFlags.scheduler = rtsTrue; + break; + case 'i': + RtsFlags.DebugFlags.interpreter = rtsTrue; + break; + case 'w': + RtsFlags.DebugFlags.weak = rtsTrue; + break; + case 'G': + RtsFlags.DebugFlags.gccafs = rtsTrue; + break; + case 'g': + RtsFlags.DebugFlags.gc = rtsTrue; + break; + case 'b': + RtsFlags.DebugFlags.block_alloc = rtsTrue; + break; + case 'S': + RtsFlags.DebugFlags.sanity = rtsTrue; + break; + case 't': + RtsFlags.DebugFlags.stable = rtsTrue; + break; + case 'p': + RtsFlags.DebugFlags.prof = rtsTrue; + break; + case 'l': + RtsFlags.DebugFlags.linker = rtsTrue; + break; + case 'a': + RtsFlags.DebugFlags.apply = rtsTrue; + break; + case 'm': + RtsFlags.DebugFlags.stm = rtsTrue; + break; + case 'z': + RtsFlags.DebugFlags.squeeze = rtsTrue; + break; + case 'c': + RtsFlags.DebugFlags.hpc = rtsTrue; + break; + case 'r': + RtsFlags.DebugFlags.sparks = rtsTrue; + break; + default: + bad_option( arg ); + } + } + // -Dx also turns on -v. Use -l to direct trace + // events to the .eventlog file instead. + RtsFlags.TraceFlags.tracing = TRACE_STDERR; +} +#endif + +#ifdef USE_PAPI +static void read_papi_flags(const char *arg) +{ + // Already parsed "-a" + switch(arg[2]) { + case '1': + RtsFlags.PapiFlags.eventType = PAPI_FLAG_CACHE_L1; + break; + case '2': + RtsFlags.PapiFlags.eventType = PAPI_FLAG_CACHE_L2; + break; + case 'b': + RtsFlags.PapiFlags.eventType = PAPI_FLAG_BRANCH; + break; + case 's': + RtsFlags.PapiFlags.eventType = PAPI_FLAG_STALLS; + break; + case 'e': + RtsFlags.PapiFlags.eventType = PAPI_FLAG_CB_EVENTS; + break; + case '+': + case '#': + if (RtsFlags.PapiFlags.numUserEvents >= MAX_PAPI_USER_EVENTS) { + errorBelch("maximum number of PAPI events reached"); + stg_exit(EXIT_FAILURE); + } + nat eventNum = RtsFlags.PapiFlags.numUserEvents++; + char kind = arg[2]; + nat eventKind = + kind == '+' ? PAPI_PRESET_EVENT_KIND : PAPI_NATIVE_EVENT_KIND; + + RtsFlags.PapiFlags.userEvents[eventNum] = arg + 3; + RtsFlags.PapiFlags.eventType = PAPI_USER_EVENTS; + RtsFlags.PapiFlags.userEventsKind[eventNum] = eventKind; + break; + default: + bad_option( arg ); + } +} +#endif + +#ifdef PROFILING +// Parse a "-h" flag, returning whether the parse resulted in an error. +static rtsBool read_heap_profiling_flag(const char *arg) +{ + // Already parsed "-h" + rtsBool error = rtsFalse; + switch (arg[2]) { + case '\0': + case 'C': + case 'c': + case 'M': + case 'm': + case 'D': + case 'd': + case 'Y': + case 'y': + case 'R': + case 'r': + case 'B': + case 'b': + if (arg[2] != '\0' && arg[3] != '\0') { + { + char *left = strchr(arg, '{'); + char *right = strrchr(arg, '}'); + + // curly braces are optional, for + // backwards compat. + if (left) + left = left+1; + else + left = arg + 3; + + if (!right) + right = arg + strlen(arg); + + *right = '\0'; + + switch (arg[2]) { + case 'c': // cost centre label select + RtsFlags.ProfFlags.ccSelector = left; + break; + case 'C': + RtsFlags.ProfFlags.ccsSelector = left; + break; + case 'M': + case 'm': // cost centre module select + RtsFlags.ProfFlags.modSelector = left; + break; + case 'D': + case 'd': // closure descr select + RtsFlags.ProfFlags.descrSelector = left; + break; + case 'Y': + case 'y': // closure type select + RtsFlags.ProfFlags.typeSelector = left; + break; + case 'R': + case 'r': // retainer select + RtsFlags.ProfFlags.retainerSelector = left; + break; + case 'B': + case 'b': // biography select + RtsFlags.ProfFlags.bioSelector = left; + break; + } + } + break; + } + + if (RtsFlags.ProfFlags.doHeapProfile != 0) { + errorBelch("multiple heap profile options"); + error = rtsTrue; + break; + } + + switch (arg[2]) { + case '\0': + case 'C': + case 'c': + RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_CCS; + break; + case 'M': + case 'm': + RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_MOD; + break; + case 'D': + case 'd': + RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_DESCR; + break; + case 'Y': + case 'y': + RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_TYPE; + break; + case 'R': + case 'r': + RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_RETAINER; + break; + case 'B': + case 'b': + RtsFlags.ProfFlags.doHeapProfile = HEAP_BY_LDV; + break; + } + break; + + default: + errorBelch("invalid heap profile option: %s", arg); + error = rtsTrue; + } + + return error; +} +#endif + #if defined(TRACING) -static void read_trace_flags(char *arg) +static void read_trace_flags(const char *arg) { - char *c; + const char *c; rtsBool enabled = rtsTrue; /* Syntax for tracing flags currently looks like: * -- cgit v1.2.1