checker: abort prematurely on too many errors (#11185)
parent
6bc44acc70
commit
b3094b0667
|
@ -131,7 +131,6 @@ const (
|
||||||
'-w',
|
'-w',
|
||||||
'-print-v-files',
|
'-print-v-files',
|
||||||
'-error-limit',
|
'-error-limit',
|
||||||
'-warn-error-limit',
|
|
||||||
'-os',
|
'-os',
|
||||||
'-printfn',
|
'-printfn',
|
||||||
'-cflags',
|
'-cflags',
|
||||||
|
|
|
@ -82,9 +82,10 @@ NB: the build flags are shared with the run command too:
|
||||||
d) the function name
|
d) the function name
|
||||||
NB: if you want to output the profile info to stdout, use `-profile -`.
|
NB: if you want to output the profile info to stdout, use `-profile -`.
|
||||||
|
|
||||||
-warn-error-limit <limit>
|
-error-limit <limit>
|
||||||
Limit of warnings / errors that will be accumulated (defaults to 100).
|
The maximum amount of warnings / errors / notices, that will be accumulated (defaults to 100).
|
||||||
Settings this to a negative value will disable the limit.
|
The checker will abort prematurely once this limit has been reached.
|
||||||
|
Setting this to 0 or a negative value, will disable the limit.
|
||||||
|
|
||||||
-profile-no-inline
|
-profile-no-inline
|
||||||
Skip [inline] functions when profiling.
|
Skip [inline] functions when profiling.
|
||||||
|
|
|
@ -19,14 +19,13 @@ pub:
|
||||||
compiled_dir string // contains os.real_path() of the dir of the final file beeing compiled, or the dir itself when doing `v .`
|
compiled_dir string // contains os.real_path() of the dir of the final file beeing compiled, or the dir itself when doing `v .`
|
||||||
module_path string
|
module_path string
|
||||||
mut:
|
mut:
|
||||||
pref &pref.Preferences
|
pref &pref.Preferences
|
||||||
checker &checker.Checker
|
checker &checker.Checker
|
||||||
transformer &transformer.Transformer
|
transformer &transformer.Transformer
|
||||||
out_name_c string
|
out_name_c string
|
||||||
out_name_js string
|
out_name_js string
|
||||||
max_nr_errors int = 100
|
stats_lines int // size of backend generated source code in lines
|
||||||
stats_lines int // size of backend generated source code in lines
|
stats_bytes int // size of backend generated source code in bytes
|
||||||
stats_bytes int // size of backend generated source code in bytes
|
|
||||||
pub mut:
|
pub mut:
|
||||||
module_search_paths []string
|
module_search_paths []string
|
||||||
parsed_files []&ast.File
|
parsed_files []&ast.File
|
||||||
|
@ -64,10 +63,8 @@ pub fn new_builder(pref &pref.Preferences) Builder {
|
||||||
checker: checker.new_checker(table, pref)
|
checker: checker.new_checker(table, pref)
|
||||||
transformer: transformer.new_transformer(pref)
|
transformer: transformer.new_transformer(pref)
|
||||||
compiled_dir: compiled_dir
|
compiled_dir: compiled_dir
|
||||||
max_nr_errors: if pref.error_limit > 0 { pref.error_limit } else { 100 }
|
|
||||||
cached_msvc: msvc
|
cached_msvc: msvc
|
||||||
}
|
}
|
||||||
// max_nr_errors: pref.error_limit ?? 100 TODO potential syntax?
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (mut b Builder) front_stages(v_files []string) ? {
|
pub fn (mut b Builder) front_stages(v_files []string) ? {
|
||||||
|
@ -366,7 +363,7 @@ fn (b &Builder) print_warnings_and_errors() {
|
||||||
println('$b.checker.nr_notices notices')
|
println('$b.checker.nr_notices notices')
|
||||||
}
|
}
|
||||||
if b.checker.nr_notices > 0 && !b.pref.skip_warnings {
|
if b.checker.nr_notices > 0 && !b.pref.skip_warnings {
|
||||||
for i, err in b.checker.notices {
|
for err in b.checker.notices {
|
||||||
kind := if b.pref.is_verbose {
|
kind := if b.pref.is_verbose {
|
||||||
'$err.reporter notice #$b.checker.nr_notices:'
|
'$err.reporter notice #$b.checker.nr_notices:'
|
||||||
} else {
|
} else {
|
||||||
|
@ -377,13 +374,10 @@ fn (b &Builder) print_warnings_and_errors() {
|
||||||
if err.details.len > 0 {
|
if err.details.len > 0 {
|
||||||
eprintln('Details: $err.details')
|
eprintln('Details: $err.details')
|
||||||
}
|
}
|
||||||
if i > b.max_nr_errors {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if b.checker.nr_warnings > 0 && !b.pref.skip_warnings {
|
if b.checker.nr_warnings > 0 && !b.pref.skip_warnings {
|
||||||
for i, err in b.checker.warnings {
|
for err in b.checker.warnings {
|
||||||
kind := if b.pref.is_verbose {
|
kind := if b.pref.is_verbose {
|
||||||
'$err.reporter warning #$b.checker.nr_warnings:'
|
'$err.reporter warning #$b.checker.nr_warnings:'
|
||||||
} else {
|
} else {
|
||||||
|
@ -394,10 +388,6 @@ fn (b &Builder) print_warnings_and_errors() {
|
||||||
if err.details.len > 0 {
|
if err.details.len > 0 {
|
||||||
eprintln('Details: $err.details')
|
eprintln('Details: $err.details')
|
||||||
}
|
}
|
||||||
// eprintln('')
|
|
||||||
if i > b.max_nr_errors {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//
|
//
|
||||||
|
@ -405,7 +395,7 @@ fn (b &Builder) print_warnings_and_errors() {
|
||||||
println('$b.checker.nr_errors errors')
|
println('$b.checker.nr_errors errors')
|
||||||
}
|
}
|
||||||
if b.checker.nr_errors > 0 {
|
if b.checker.nr_errors > 0 {
|
||||||
for i, err in b.checker.errors {
|
for err in b.checker.errors {
|
||||||
kind := if b.pref.is_verbose {
|
kind := if b.pref.is_verbose {
|
||||||
'$err.reporter error #$b.checker.nr_errors:'
|
'$err.reporter error #$b.checker.nr_errors:'
|
||||||
} else {
|
} else {
|
||||||
|
@ -416,10 +406,6 @@ fn (b &Builder) print_warnings_and_errors() {
|
||||||
if err.details.len > 0 {
|
if err.details.len > 0 {
|
||||||
eprintln('Details: $err.details')
|
eprintln('Details: $err.details')
|
||||||
}
|
}
|
||||||
// eprintln('')
|
|
||||||
if i > b.max_nr_errors {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
b.show_total_warns_and_errors_stats()
|
b.show_total_warns_and_errors_stats()
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
|
@ -53,6 +53,7 @@ pub mut:
|
||||||
nr_errors int
|
nr_errors int
|
||||||
nr_warnings int
|
nr_warnings int
|
||||||
nr_notices int
|
nr_notices int
|
||||||
|
should_abort bool // when too many errors/warnings/notices are accumulated, .should_abort becomes true. It is checked in statement/expression loops, so the checker can return early, instead of wasting time.
|
||||||
errors []errors.Error
|
errors []errors.Error
|
||||||
warnings []errors.Warning
|
warnings []errors.Warning
|
||||||
notices []errors.Notice
|
notices []errors.Notice
|
||||||
|
@ -139,18 +140,27 @@ pub fn (mut c Checker) check(ast_file &ast.File) {
|
||||||
c.expr_level = 0
|
c.expr_level = 0
|
||||||
c.stmt(stmt)
|
c.stmt(stmt)
|
||||||
}
|
}
|
||||||
|
if c.should_abort {
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
for mut stmt in ast_file.stmts {
|
for mut stmt in ast_file.stmts {
|
||||||
if stmt is ast.GlobalDecl {
|
if stmt is ast.GlobalDecl {
|
||||||
c.expr_level = 0
|
c.expr_level = 0
|
||||||
c.stmt(stmt)
|
c.stmt(stmt)
|
||||||
}
|
}
|
||||||
|
if c.should_abort {
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
for mut stmt in ast_file.stmts {
|
for mut stmt in ast_file.stmts {
|
||||||
if stmt !is ast.ConstDecl && stmt !is ast.GlobalDecl && stmt !is ast.ExprStmt {
|
if stmt !is ast.ConstDecl && stmt !is ast.GlobalDecl && stmt !is ast.ExprStmt {
|
||||||
c.expr_level = 0
|
c.expr_level = 0
|
||||||
c.stmt(stmt)
|
c.stmt(stmt)
|
||||||
}
|
}
|
||||||
|
if c.should_abort {
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
c.check_scope_vars(c.file.scope)
|
c.check_scope_vars(c.file.scope)
|
||||||
}
|
}
|
||||||
|
@ -7593,7 +7603,8 @@ fn (c &Checker) check_struct_signature(from ast.Struct, to ast.Struct) bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (mut c Checker) note(message string, pos token.Position) {
|
pub fn (mut c Checker) note(message string, pos token.Position) {
|
||||||
if c.nr_notices >= c.pref.warn_error_limit && c.pref.warn_error_limit >= 0 {
|
if c.pref.message_limit >= 0 && c.nr_notices >= c.pref.message_limit {
|
||||||
|
c.should_abort = true
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
mut details := ''
|
mut details := ''
|
||||||
|
@ -7625,7 +7636,8 @@ fn (mut c Checker) warn_or_error(message string, pos token.Position, warn bool)
|
||||||
}
|
}
|
||||||
if warn && !c.pref.skip_warnings {
|
if warn && !c.pref.skip_warnings {
|
||||||
c.nr_warnings++
|
c.nr_warnings++
|
||||||
if c.nr_warnings >= c.pref.warn_error_limit && c.pref.warn_error_limit >= 0 {
|
if c.pref.message_limit >= 0 && c.nr_warnings >= c.pref.message_limit {
|
||||||
|
c.should_abort = true
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
wrn := errors.Warning{
|
wrn := errors.Warning{
|
||||||
|
@ -7644,19 +7656,21 @@ fn (mut c Checker) warn_or_error(message string, pos token.Position, warn bool)
|
||||||
exit(1)
|
exit(1)
|
||||||
}
|
}
|
||||||
c.nr_errors++
|
c.nr_errors++
|
||||||
if c.errors.len < c.pref.warn_error_limit || c.pref.warn_error_limit < 0 {
|
if c.pref.message_limit >= 0 && c.errors.len >= c.pref.message_limit {
|
||||||
if pos.line_nr !in c.error_lines {
|
c.should_abort = true
|
||||||
err := errors.Error{
|
return
|
||||||
reporter: errors.Reporter.checker
|
}
|
||||||
pos: pos
|
if pos.line_nr !in c.error_lines {
|
||||||
file_path: c.file.path
|
err := errors.Error{
|
||||||
message: message
|
reporter: errors.Reporter.checker
|
||||||
details: details
|
pos: pos
|
||||||
}
|
file_path: c.file.path
|
||||||
c.file.errors << err
|
message: message
|
||||||
c.errors << err
|
details: details
|
||||||
c.error_lines << pos.line_nr
|
|
||||||
}
|
}
|
||||||
|
c.file.errors << err
|
||||||
|
c.errors << err
|
||||||
|
c.error_lines << pos.line_nr
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -79,6 +79,7 @@ mut:
|
||||||
inside_defer bool
|
inside_defer bool
|
||||||
comp_if_cond bool
|
comp_if_cond bool
|
||||||
defer_vars []ast.Ident
|
defer_vars []ast.Ident
|
||||||
|
should_abort bool // when too many errors/warnings/notices are accumulated, should_abort becomes true, and the parser should stop
|
||||||
}
|
}
|
||||||
|
|
||||||
// for tests
|
// for tests
|
||||||
|
@ -267,6 +268,9 @@ pub fn (mut p Parser) parse() &ast.File {
|
||||||
p.attrs = []
|
p.attrs = []
|
||||||
}
|
}
|
||||||
stmts << stmt
|
stmts << stmt
|
||||||
|
if p.should_abort {
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
p.scope.end_pos = p.tok.pos
|
p.scope.end_pos = p.tok.pos
|
||||||
return &ast.File{
|
return &ast.File{
|
||||||
|
@ -579,6 +583,9 @@ pub fn (mut p Parser) top_stmt() ast.Stmt {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if p.should_abort {
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// TODO remove dummy return statement
|
// TODO remove dummy return statement
|
||||||
// the compiler complains if it's not there
|
// the compiler complains if it's not there
|
||||||
|
@ -1649,9 +1656,11 @@ pub fn (mut p Parser) error_with_error(error errors.Error) {
|
||||||
eprintln(ferror)
|
eprintln(ferror)
|
||||||
exit(1)
|
exit(1)
|
||||||
} else {
|
} else {
|
||||||
if p.errors.len < p.pref.warn_error_limit || p.pref.warn_error_limit < 0 {
|
if p.pref.message_limit >= 0 && p.errors.len >= p.pref.message_limit {
|
||||||
p.errors << error
|
p.should_abort = true
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
p.errors << error
|
||||||
}
|
}
|
||||||
if p.pref.output_mode == .silent {
|
if p.pref.output_mode == .silent {
|
||||||
// Normally, parser errors mean that the parser exits immediately, so there can be only 1 parser error.
|
// Normally, parser errors mean that the parser exits immediately, so there can be only 1 parser error.
|
||||||
|
@ -1674,13 +1683,15 @@ pub fn (mut p Parser) warn_with_pos(s string, pos token.Position) {
|
||||||
ferror := util.formatted_error('warning:', s, p.file_name, pos)
|
ferror := util.formatted_error('warning:', s, p.file_name, pos)
|
||||||
eprintln(ferror)
|
eprintln(ferror)
|
||||||
} else {
|
} else {
|
||||||
if p.warnings.len < p.pref.warn_error_limit || p.pref.warn_error_limit < 0 {
|
if p.pref.message_limit >= 0 && p.warnings.len >= p.pref.message_limit {
|
||||||
p.warnings << errors.Warning{
|
p.should_abort = true
|
||||||
file_path: p.file_name
|
return
|
||||||
pos: pos
|
}
|
||||||
reporter: .parser
|
p.warnings << errors.Warning{
|
||||||
message: s
|
file_path: p.file_name
|
||||||
}
|
pos: pos
|
||||||
|
reporter: .parser
|
||||||
|
message: s
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -176,7 +176,6 @@ pub mut:
|
||||||
no_std bool // when true, do not pass -std=c99 to the C backend
|
no_std bool // when true, do not pass -std=c99 to the C backend
|
||||||
use_color ColorOutput // whether the warnings/errors should use ANSI color escapes.
|
use_color ColorOutput // whether the warnings/errors should use ANSI color escapes.
|
||||||
is_parallel bool
|
is_parallel bool
|
||||||
error_limit int
|
|
||||||
is_vweb bool // skip _ var warning in templates
|
is_vweb bool // skip _ var warning in templates
|
||||||
only_check_syntax bool // when true, just parse the files, then stop, before running checker
|
only_check_syntax bool // when true, just parse the files, then stop, before running checker
|
||||||
experimental bool // enable experimental features
|
experimental bool // enable experimental features
|
||||||
|
@ -191,7 +190,7 @@ pub mut:
|
||||||
gc_mode GarbageCollectionMode = .no_gc // .no_gc, .boehm, .boehm_leak, ...
|
gc_mode GarbageCollectionMode = .no_gc // .no_gc, .boehm, .boehm_leak, ...
|
||||||
is_cstrict bool // turn on more C warnings; slightly slower
|
is_cstrict bool // turn on more C warnings; slightly slower
|
||||||
assert_failure_mode AssertFailureMode // whether to call abort() or print_backtrace() after an assertion failure
|
assert_failure_mode AssertFailureMode // whether to call abort() or print_backtrace() after an assertion failure
|
||||||
warn_error_limit int = 100 // limit of warnings/errors to be accumulated
|
message_limit int = 100 // the maximum amount of warnings/errors/notices that will be accumulated
|
||||||
// checker settings:
|
// checker settings:
|
||||||
checker_match_exhaustive_cutoff_limit int = 12
|
checker_match_exhaustive_cutoff_limit int = 12
|
||||||
}
|
}
|
||||||
|
@ -495,9 +494,6 @@ pub fn parse_args(known_external_commands []string, args []string) (&Preferences
|
||||||
'-print-v-files' {
|
'-print-v-files' {
|
||||||
res.print_v_files = true
|
res.print_v_files = true
|
||||||
}
|
}
|
||||||
'-error-limit' {
|
|
||||||
res.error_limit = cmdline.option(current_args, '-error-limit', '0').int()
|
|
||||||
}
|
|
||||||
'-os' {
|
'-os' {
|
||||||
target_os := cmdline.option(current_args, '-os', '')
|
target_os := cmdline.option(current_args, '-os', '')
|
||||||
i++
|
i++
|
||||||
|
@ -528,8 +524,8 @@ pub fn parse_args(known_external_commands []string, args []string) (&Preferences
|
||||||
}
|
}
|
||||||
i++
|
i++
|
||||||
}
|
}
|
||||||
'-warn-error-limit' {
|
'-message-limit' {
|
||||||
res.warn_error_limit = cmdline.option(current_args, arg, '10').int()
|
res.message_limit = cmdline.option(current_args, arg, '5').int()
|
||||||
i++
|
i++
|
||||||
}
|
}
|
||||||
'-cc' {
|
'-cc' {
|
||||||
|
|
|
@ -57,6 +57,7 @@ pub mut:
|
||||||
warnings []errors.Warning
|
warnings []errors.Warning
|
||||||
notices []errors.Notice
|
notices []errors.Notice
|
||||||
vet_errors []vet.Error
|
vet_errors []vet.Error
|
||||||
|
should_abort bool // when too many errors/warnings/notices are accumulated, should_abort becomes true, and the scanner should stop
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -564,7 +565,7 @@ pub fn (mut s Scanner) scan_remaining_text() {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
s.all_tokens << t
|
s.all_tokens << t
|
||||||
if t.kind == .eof {
|
if t.kind == .eof || s.should_abort {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -579,7 +580,7 @@ pub fn (mut s Scanner) buffer_scan() token.Token {
|
||||||
for {
|
for {
|
||||||
cidx := s.tidx
|
cidx := s.tidx
|
||||||
s.tidx++
|
s.tidx++
|
||||||
if cidx >= s.all_tokens.len {
|
if cidx >= s.all_tokens.len || s.should_abort {
|
||||||
return s.end_of_file()
|
return s.end_of_file()
|
||||||
}
|
}
|
||||||
if s.all_tokens[cidx].kind == .comment {
|
if s.all_tokens[cidx].kind == .comment {
|
||||||
|
@ -635,7 +636,7 @@ fn (mut s Scanner) text_scan() token.Token {
|
||||||
if !s.is_inside_string {
|
if !s.is_inside_string {
|
||||||
s.skip_whitespace()
|
s.skip_whitespace()
|
||||||
}
|
}
|
||||||
if s.pos >= s.text.len {
|
if s.pos >= s.text.len || s.should_abort {
|
||||||
return s.end_of_file()
|
return s.end_of_file()
|
||||||
}
|
}
|
||||||
// End of $var, start next string
|
// End of $var, start next string
|
||||||
|
@ -1354,13 +1355,15 @@ pub fn (mut s Scanner) warn(msg string) {
|
||||||
if s.pref.output_mode == .stdout {
|
if s.pref.output_mode == .stdout {
|
||||||
eprintln(util.formatted_error('warning:', msg, s.file_path, pos))
|
eprintln(util.formatted_error('warning:', msg, s.file_path, pos))
|
||||||
} else {
|
} else {
|
||||||
if s.warnings.len < s.pref.warn_error_limit || s.pref.warn_error_limit < 0 {
|
if s.pref.message_limit >= 0 && s.warnings.len >= s.pref.message_limit {
|
||||||
s.warnings << errors.Warning{
|
s.should_abort = true
|
||||||
file_path: s.file_path
|
return
|
||||||
pos: pos
|
}
|
||||||
reporter: .scanner
|
s.warnings << errors.Warning{
|
||||||
message: msg
|
file_path: s.file_path
|
||||||
}
|
pos: pos
|
||||||
|
reporter: .scanner
|
||||||
|
message: msg
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1378,13 +1381,15 @@ pub fn (mut s Scanner) error(msg string) {
|
||||||
if s.pref.fatal_errors {
|
if s.pref.fatal_errors {
|
||||||
exit(1)
|
exit(1)
|
||||||
}
|
}
|
||||||
if s.errors.len < s.pref.warn_error_limit || s.pref.warn_error_limit < 0 {
|
if s.pref.message_limit >= 0 && s.errors.len >= s.pref.message_limit {
|
||||||
s.errors << errors.Error{
|
s.should_abort = true
|
||||||
file_path: s.file_path
|
return
|
||||||
pos: pos
|
}
|
||||||
reporter: .scanner
|
s.errors << errors.Error{
|
||||||
message: msg
|
file_path: s.file_path
|
||||||
}
|
pos: pos
|
||||||
|
reporter: .scanner
|
||||||
|
message: msg
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue