mirror of
https://github.com/luau-lang/luau.git
synced 2025-01-08 12:29:09 +00:00
c2ba1058c3
# What's changed? - Record the location of properties for table types (closes #802) - Implement stricter UTF-8 validations as per the RFC (https://github.com/luau-lang/rfcs/pull/1) - Implement `buffer` as a new type in both the old and new solvers. - Changed errors produced by some `buffer` builtins to be a bit more generic to avoid platform-dependent error messages. - Fixed a bug where `Unifier` would copy some persistent types, tripping some internal assertions. - Type checking rules on relational operators is now a little bit more lax. - Improve dead code elimination for some `if` statements with complex always-false conditions ## New type solver - Dataflow analysis now generates phi nodes on exit of branches. - Dataflow analysis avoids producing a new definition for locals or properties that are not owned by that loop. - If a function parameter has been constrained to `never`, report errors at all uses of that parameter within that function. - Switch to using the new `Luau::Set` to replace `std::unordered_set` to alleviate some poor allocation characteristics which was negatively affecting overall performance. - Subtyping can now report many failing reasons instead of just the first one that we happened to find during the test. - Subtyping now also report reasons for type pack mismatches. - When visiting `if` statements or expressions, the resulting context are the common terms in both branches. ## Native codegen - Implement support for `buffer` builtins to its IR for x64 and A64. - Optimized `table.insert` by not inserting a table barrier if it is fastcalled with a constant. ## Internal Contributors Co-authored-by: Aaron Weiss <aaronweiss@roblox.com> Co-authored-by: Alexander McCord <amccord@roblox.com> Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Arseny Kapoulkine <arseny@roblox.com> Co-authored-by: Aviral Goel <agoel@roblox.com> Co-authored-by: Lily Brown <lbrown@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com>
103 lines
3.3 KiB
Python
103 lines
3.3 KiB
Python
#!/usr/bin/python3
|
|
# This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
# Given a fuzzer binary and a list of crashing programs, this tool collects unique crash reasons and prints reproducers.
|
|
|
|
import argparse
|
|
import multiprocessing
|
|
import os
|
|
import re
|
|
import subprocess
|
|
import sys
|
|
|
|
|
|
class Reproducer:
|
|
def __init__(self, file, reason, fingerprint):
|
|
self.file = file
|
|
self.reason = reason
|
|
self.fingerprint = fingerprint
|
|
|
|
|
|
def get_crash_reason(binary, file, remove_passing):
|
|
res = subprocess.run(
|
|
[binary, file], stdout=subprocess.DEVNULL, stderr=subprocess.PIPE)
|
|
|
|
if res.returncode == 0:
|
|
if remove_passing:
|
|
print(f"Warning: {binary} {file} returned 0; removing from result set.", file=sys.stderr)
|
|
os.remove(file)
|
|
else:
|
|
print(f"Warning: {binary} {file} returned 0", file=sys.stderr)
|
|
|
|
return None
|
|
|
|
err = res.stderr.decode("utf-8")
|
|
|
|
if (pos := err.find("ERROR: AddressSanitizer:")) != -1:
|
|
return err[pos:]
|
|
|
|
if (pos := err.find("ERROR: libFuzzer:")) != -1:
|
|
return err[pos:]
|
|
|
|
print(f"Warning: {binary} {file} returned unrecognized error {err}", file=sys.stderr)
|
|
return None
|
|
|
|
|
|
def get_crash_fingerprint(reason):
|
|
# Due to ASLR addresses are different every time, so we filter them out
|
|
reason = re.sub(r"0x[0-9a-f]+", "0xXXXX", reason)
|
|
return reason
|
|
|
|
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("binary")
|
|
parser.add_argument("files", action="append", default=[])
|
|
parser.add_argument("--remove-duplicates", action="store_true")
|
|
parser.add_argument("--remove-passing", action="store_true")
|
|
parser.add_argument("--workers", action="store", default=1, type=int)
|
|
parser.add_argument("--verbose", "-v", action="count", default=0, dest="verbosity")
|
|
|
|
args = parser.parse_args()
|
|
|
|
def process_file(file):
|
|
reason = get_crash_reason(args.binary, file, args.remove_passing)
|
|
if reason is None:
|
|
return None
|
|
|
|
fingerprint = get_crash_fingerprint(reason)
|
|
return Reproducer(file, reason, fingerprint)
|
|
|
|
|
|
filter_targets = []
|
|
if len(args.files) == 1:
|
|
for root, dirs, files in os.walk(args.files[0]):
|
|
for file in files:
|
|
filter_targets.append(os.path.join(root, file))
|
|
else:
|
|
filter_targets = args.files
|
|
|
|
with multiprocessing.Pool(processes = args.workers) as pool:
|
|
print(f"Processing {len(filter_targets)} reproducers across {args.workers} workers.")
|
|
reproducers = [r for r in pool.map(process_file, filter_targets) if r is not None]
|
|
|
|
seen = set()
|
|
for index, reproducer in enumerate(reproducers):
|
|
if reproducer.fingerprint in seen:
|
|
if sys.stdout.isatty():
|
|
print("-\|/"[index % 4], end="\r")
|
|
|
|
if args.remove_duplicates:
|
|
if args.verbosity >= 1:
|
|
print(f"Removing duplicate reducer {reproducer.file}.")
|
|
os.remove(reproducer.file)
|
|
|
|
continue
|
|
|
|
seen.add(reproducer.fingerprint)
|
|
if args.verbosity >= 2:
|
|
print(f"Reproducer: {args.binary} {reproducer.file}")
|
|
print(f"Output: {reproducer.reason}")
|
|
|
|
print(f"Total unique crashes: {len(seen)}")
|
|
if args.remove_duplicates:
|
|
print(f"Duplicate reproducers have been removed.")
|