2018-03-07 08:54:53 +01:00
|
|
|
# Copyright 2018 the V8 project authors. All rights reserved.
|
|
|
|
# Use of this source code is governed by a BSD-style license that can be
|
|
|
|
# found in the LICENSE file.
|
|
|
|
|
|
|
|
import re
|
|
|
|
|
2022-04-19 09:00:36 +02:00
|
|
|
from itertools import zip_longest
|
2022-04-12 11:10:15 +02:00
|
|
|
|
2018-03-07 08:54:53 +01:00
|
|
|
from . import base
|
|
|
|
|
|
|
|
|
2021-07-14 11:30:07 +02:00
|
|
|
class OutProc(base.ExpectedOutProc):
|
|
|
|
def __init__(self, expected_outcomes, basepath, expected_fail,
|
|
|
|
expected_filename, regenerate_expected_files):
|
|
|
|
super(OutProc, self).__init__(expected_outcomes, expected_filename,
|
|
|
|
regenerate_expected_files)
|
2018-03-07 08:54:53 +01:00
|
|
|
self._basepath = basepath
|
|
|
|
self._expected_fail = expected_fail
|
|
|
|
|
|
|
|
def _is_failure_output(self, output):
|
|
|
|
fail = output.exit_code != 0
|
|
|
|
if fail != self._expected_fail:
|
|
|
|
return True
|
|
|
|
|
|
|
|
expected_lines = []
|
|
|
|
# Can't use utils.ReadLinesFrom() here because it strips whitespace.
|
2023-10-05 08:46:07 +02:00
|
|
|
with open(self._basepath.with_suffix('.out')) as f:
|
2018-03-07 08:54:53 +01:00
|
|
|
for line in f:
|
|
|
|
if line.startswith("#") or not line.strip():
|
|
|
|
continue
|
|
|
|
expected_lines.append(line)
|
|
|
|
raw_lines = output.stdout.splitlines()
|
|
|
|
actual_lines = [ s for s in raw_lines if not self._ignore_line(s) ]
|
|
|
|
if len(expected_lines) != len(actual_lines):
|
|
|
|
return True
|
|
|
|
|
2019-08-16 11:32:46 +02:00
|
|
|
# Try .js first, and fall back to .mjs.
|
|
|
|
# TODO(v8:9406): clean this up by never separating the path from
|
|
|
|
# the extension in the first place.
|
2023-10-05 08:46:07 +02:00
|
|
|
base_path = self._basepath.with_suffix('.js')
|
|
|
|
if not base_path.exists():
|
|
|
|
base_path = self._basepath.with_suffix('.mjs')
|
2019-08-16 11:32:46 +02:00
|
|
|
|
2018-03-07 08:54:53 +01:00
|
|
|
env = {
|
2023-10-05 08:46:07 +02:00
|
|
|
'basename': base_path.name,
|
2018-03-07 08:54:53 +01:00
|
|
|
}
|
2022-04-12 11:10:15 +02:00
|
|
|
for (expected, actual) in zip_longest(
|
2018-03-07 08:54:53 +01:00
|
|
|
expected_lines, actual_lines, fillvalue=''):
|
|
|
|
pattern = re.escape(expected.rstrip() % env)
|
|
|
|
pattern = pattern.replace('\\*', '.*')
|
2025-04-29 08:03:15 +02:00
|
|
|
pattern = pattern.replace('\\{NUMBER\\}', r'-?\d+(?:\.\d*)?')
|
2023-10-05 08:46:07 +02:00
|
|
|
# Note: The character sequence for printing an address in C++ is
|
|
|
|
# implementation defined.
|
|
|
|
pattern = pattern.replace('\\{ADDRESS\\}', r'(0x)?[0-9A-Fa-f]+')
|
2018-03-07 08:54:53 +01:00
|
|
|
pattern = '^%s$' % pattern
|
|
|
|
if not re.match(pattern, actual):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _ignore_line(self, string):
|
|
|
|
"""Ignore empty lines, valgrind output, Android output."""
|
|
|
|
return (
|
2023-10-05 08:46:07 +02:00
|
|
|
not string or #
|
|
|
|
not string.strip() or #
|
|
|
|
string.startswith("==") or #
|
|
|
|
string.startswith("**") or #
|
|
|
|
string.startswith("ANDROID") or #
|
|
|
|
# Android linker warning.
|
|
|
|
string.startswith('WARNING: linker:') or #
|
|
|
|
# Testing on Android devices mixes stderr into stdout.
|
|
|
|
string ==
|
|
|
|
"V8 is running with experimental features enabled. Stability and security will suffer."
|
2024-08-14 20:41:00 +02:00
|
|
|
or #
|
|
|
|
string == "Concurrent maglev has been disabled for tracing.")
|