Donate cpu server crash locations (#2276)

donate-cpu-server: Print grouped stack traces in crash report

Example output:

	Packages: psi xenomai
	Token::hasKnownValue (this=0x0) at lib/token.h:988
	988	        return mImpl->mValues && std::any_of(mImpl->mValues->begin(), mImpl->mValues->end(), std::mem_fn(&ValueFlow::Value::isKnown));
	#0 Token::hasKnownValue (...) at lib/token.h:988
	#1 valueFlowReverse (...) at build/valueflow.cpp:3775
	#2 valueFlowBeforeCondition (...) at /usr/include/c++/8/bits/stl_list.h:301
	#3 ValueFlow::setValues (...) at build/valueflow.cpp:8403
	#4 Tokenizer::simplifyTokens1 (...) at build/tokenize.cpp:11095
	#5 CppCheck::checkFile (...) at build/cppcheck.cpp:513
	#6 CppCheck::check (...) at /usr/include/c++/8/bits/basic_string.h:936
	#7 CppCheckExecutor::check_internal (...) at cli/cppcheckexecutor.cpp:884
	#8 CppCheckExecutor::check (...) at cli/cppcheckexecutor.cpp:198
	#9 main (...) at cli/main.cpp:95

	Packages: broker
	valueFlowReverse (tokenlist=tokenlist@entry=0x7fffffffbce0, tok=tok@entry=0x555555cc9930, varToken=varToken@entry=0x555555cc9b70, val=..., val2=..., errorLogger=errorLogger@entry=0x7fffffffcb20, settings=0x7fffffffcd40) at build/valueflow.cpp:3775
	3775	                if (!assignTok->hasKnownValue()) {
	#0 valueFlowReverse (...) at build/valueflow.cpp:3775
	#1 valueFlowBeforeCondition (...) at build/valueflow.cpp:4092
	#2 ValueFlow::setValues (...) at build/valueflow.cpp:8406
	#3 Tokenizer::simplifyTokens1 (...) at build/tokenize.cpp:11095
	#4 CppCheck::checkFile (...) at build/cppcheck.cpp:513
	#5 CppCheck::check (...) at build/cppcheck.cpp:197
	#6 CppCheckExecutor::check_internal (...) at cli/cppcheckexecutor.cpp:884
	#7 CppCheckExecutor::check (...) at cli/cppcheckexecutor.cpp:198
	#8 main (...) at cli/main.cpp:95
This commit is contained in:
Rikard Falkeborn 2019-10-18 09:56:15 +02:00 committed by Sebastian
parent 3a0a0fdefb
commit 6e1c7e48b6
1 changed files with 66 additions and 32 deletions

View File

@ -18,7 +18,7 @@ import operator
# Version scheme (MAJOR.MINOR.PATCH) should orientate on "Semantic Versioning" https://semver.org/ # Version scheme (MAJOR.MINOR.PATCH) should orientate on "Semantic Versioning" https://semver.org/
# Every change in this script should result in increasing the version number accordingly (exceptions may be cosmetic # Every change in this script should result in increasing the version number accordingly (exceptions may be cosmetic
# changes) # changes)
SERVER_VERSION = "1.1.6" SERVER_VERSION = "1.1.7"
OLD_VERSION = '1.89' OLD_VERSION = '1.89'
@ -76,7 +76,7 @@ def overviewReport():
return html return html
def fmt(a, b, c, d, e): def fmt(a, b, c=None, d=None, e=None, link=True):
column_width = [40, 10, 5, 6, 6, 8] column_width = [40, 10, 5, 6, 6, 8]
ret = a ret = a
while len(ret) < column_width[0]: while len(ret) < column_width[0]:
@ -86,6 +86,7 @@ def fmt(a, b, c, d, e):
while len(ret) < (column_width[0] + 1 + column_width[1]): while len(ret) < (column_width[0] + 1 + column_width[1]):
ret += ' ' ret += ' '
ret += ' ' ret += ' '
if len(b) > 10:
ret += b[-5:].rjust(column_width[2]) + ' ' ret += b[-5:].rjust(column_width[2]) + ' '
if not c is None: if not c is None:
ret += c.rjust(column_width[3]) + ' ' ret += c.rjust(column_width[3]) + ' '
@ -93,7 +94,7 @@ def fmt(a, b, c, d, e):
ret += d.rjust(column_width[4]) + ' ' ret += d.rjust(column_width[4]) + ' '
if not e is None: if not e is None:
ret += e.rjust(column_width[5]) ret += e.rjust(column_width[5])
if a != 'Package': if link:
pos = ret.find(' ') pos = ret.find(' ')
ret = '<a href="' + a + '">' + a + '</a>' + ret[pos:] ret = '<a href="' + a + '">' + a + '</a>' + ret[pos:]
return ret return ret
@ -102,7 +103,7 @@ def fmt(a, b, c, d, e):
def latestReport(latestResults): def latestReport(latestResults):
html = '<html><head><title>Latest daca@home results</title></head><body>\n' html = '<html><head><title>Latest daca@home results</title></head><body>\n'
html += '<h1>Latest daca@home results</h1>\n' html += '<h1>Latest daca@home results</h1>\n'
html += '<pre>\n<b>' + fmt('Package', 'Date Time', OLD_VERSION, 'Head', 'Diff') + '</b>\n' html += '<pre>\n<b>' + fmt('Package', 'Date Time', OLD_VERSION, 'Head', 'Diff', link=False) + '</b>\n'
# Write report for latest results # Write report for latest results
for filename in latestResults: for filename in latestResults:
@ -142,13 +143,15 @@ def crashReport(results_path):
html = '<html><head><title>Crash report</title></head><body>\n' html = '<html><head><title>Crash report</title></head><body>\n'
html += '<h1>Crash report</h1>\n' html += '<h1>Crash report</h1>\n'
html += '<pre>\n' html += '<pre>\n'
html += '<b>' + fmt('Package', 'Date Time', OLD_VERSION, 'Head', None) + '</b>\n' html += '<b>' + fmt('Package', 'Date Time', OLD_VERSION, 'Head', link=False) + '</b>\n'
current_year = datetime.date.today().year current_year = datetime.date.today().year
stack_traces = {}
for filename in sorted(glob.glob(os.path.expanduser(results_path + '/*'))): for filename in sorted(glob.glob(os.path.expanduser(results_path + '/*'))):
if not os.path.isfile(filename): if not os.path.isfile(filename):
continue continue
datestr = '' datestr = ''
for line in open(filename, 'rt'): with open(filename, 'rt') as file:
for line in file:
line = line.strip() line = line.strip()
if line.startswith('cppcheck: '): if line.startswith('cppcheck: '):
if OLD_VERSION not in line: if OLD_VERSION not in line:
@ -159,8 +162,7 @@ def crashReport(results_path):
continue continue
if line.startswith(str(current_year) + '-') or line.startswith(str(current_year - 1) + '-'): if line.startswith(str(current_year) + '-') or line.startswith(str(current_year - 1) + '-'):
datestr = line datestr = line
if not line.startswith('count:'): if line.startswith('count:'):
continue
if line.find('Crash') < 0: if line.find('Crash') < 0:
break break
package = filename[filename.rfind('/')+1:] package = filename[filename.rfind('/')+1:]
@ -171,8 +173,40 @@ def crashReport(results_path):
c1 = '' c1 = ''
if counts[1] == 'Crash!': if counts[1] == 'Crash!':
c1 = 'Crash' c1 = 'Crash'
html += fmt(package, datestr, c2, c1, None) + '\n' html += fmt(package, datestr, c2, c1) + '\n'
if c1 != 'Crash':
break break
if line.find(' received signal ') != -1:
crash_line = next(file, '').strip()
location_index = crash_line.rindex(' at ')
if location_index > 0:
code_line = next(file, '').strip();
stack_trace = []
while True:
l = next(file, '')
m = re.search(r'(?P<number>#\d+) .* (?P<function>.+)\(.*\) at (?P<location>.*)$', l)
if not m:
break
stack_trace.append(m.group('number') + ' ' + m.group('function') + '(...) at ' + m.group('location'))
key = hash(' '.join(stack_trace))
if key in stack_traces:
stack_traces[key]['code_line'] = code_line
stack_traces[key]['stack_trace'] = stack_trace
stack_traces[key]['n'] += 1
stack_traces[key]['packages'].append(package)
else:
stack_traces[key] = {'stack_trace': stack_trace, 'n': 1, 'code_line': code_line, 'packages': [package], 'crash_line': crash_line}
break
html += '</pre>\n'
html += '<pre>\n'
html += '<b>Stack traces</b>\n'
for stack_trace in sorted(stack_traces.values(), key=lambda x: x['n'], reverse=True):
html += 'Packages: ' + ' '.join(['<a href="' + p + '">' + p + '</a>' for p in stack_trace['packages']]) + '\n'
html += stack_trace['crash_line'] + '\n'
html += stack_trace['code_line'] + '\n'
html += '\n'.join(stack_trace['stack_trace']) + '\n\n'
html += '</pre>\n' html += '</pre>\n'
html += '</body></html>\n' html += '</body></html>\n'
@ -183,7 +217,7 @@ def staleReport(results_path):
html = '<html><head><title>Stale report</title></head><body>\n' html = '<html><head><title>Stale report</title></head><body>\n'
html += '<h1>Stale report</h1>\n' html += '<h1>Stale report</h1>\n'
html += '<pre>\n' html += '<pre>\n'
html += '<b>' + fmt('Package', 'Date Time', None, None, None) + '</b>\n' html += '<b>' + fmt('Package', 'Date Time', link=False) + '</b>\n'
current_year = datetime.date.today().year current_year = datetime.date.today().year
for filename in sorted(glob.glob(os.path.expanduser(results_path + '/*'))): for filename in sorted(glob.glob(os.path.expanduser(results_path + '/*'))):
if not os.path.isfile(filename): if not os.path.isfile(filename):
@ -199,7 +233,7 @@ def staleReport(results_path):
if diff.days < 30: if diff.days < 30:
continue continue
package = filename[filename.rfind('/')+1:] package = filename[filename.rfind('/')+1:]
html += fmt(package, datestr, None, None, None) + '\n' html += fmt(package, datestr) + '\n'
break break
html += '</pre>\n' html += '</pre>\n'