2017-12-29 20:43:29 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
2018-03-29 10:18:47 +02:00
|
|
|
from __future__ import print_function, division, absolute_import
|
|
|
|
|
2018-10-18 06:12:20 +02:00
|
|
|
import sys, os, subprocess, tempfile, threading
|
|
|
|
|
|
|
|
|
2018-10-19 07:54:21 +02:00
|
|
|
def which(program):
|
|
|
|
# https://stackoverflow.com/a/377028
|
|
|
|
def is_exe(fpath):
|
|
|
|
return os.path.isfile(fpath) and os.access(fpath, os.X_OK)
|
|
|
|
|
|
|
|
fpath, _ = os.path.split(program)
|
|
|
|
if fpath:
|
|
|
|
if is_exe(program):
|
|
|
|
return program
|
|
|
|
else:
|
|
|
|
for path in os.environ["PATH"].split(os.pathsep):
|
|
|
|
exe_file = os.path.join(path, program)
|
|
|
|
if is_exe(exe_file):
|
|
|
|
return exe_file
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2018-10-18 06:12:20 +02:00
|
|
|
def cmd(command):
|
|
|
|
# https://stackoverflow.com/a/4408409
|
|
|
|
# https://stackoverflow.com/a/10012262
|
|
|
|
with tempfile.TemporaryFile() as tempf:
|
|
|
|
p = subprocess.Popen (command, stderr=tempf)
|
|
|
|
is_killed = {'value': False}
|
|
|
|
|
|
|
|
def timeout(p, is_killed):
|
|
|
|
is_killed['value'] = True
|
|
|
|
p.kill()
|
|
|
|
timer = threading.Timer (2, timeout, [p, is_killed])
|
|
|
|
|
|
|
|
try:
|
|
|
|
timer.start()
|
|
|
|
p.wait ()
|
|
|
|
tempf.seek (0)
|
|
|
|
text = tempf.read().decode ("utf-8").strip ()
|
|
|
|
returncode = p.returncode
|
|
|
|
finally:
|
|
|
|
timer.cancel()
|
|
|
|
|
|
|
|
if is_killed['value']:
|
|
|
|
text = 'error: timeout, ' + text
|
|
|
|
returncode = 1
|
|
|
|
|
|
|
|
return text, returncode
|
|
|
|
|
2017-12-29 20:43:29 +01:00
|
|
|
|
|
|
|
srcdir = os.environ.get ("srcdir", ".")
|
|
|
|
EXEEXT = os.environ.get ("EXEEXT", "")
|
|
|
|
top_builddir = os.environ.get ("top_builddir", ".")
|
2018-03-15 21:04:31 +01:00
|
|
|
hb_shape_fuzzer = os.path.join (top_builddir, "hb-shape-fuzzer" + EXEEXT)
|
2018-01-01 08:47:51 +01:00
|
|
|
|
2018-03-15 21:04:31 +01:00
|
|
|
if not os.path.exists (hb_shape_fuzzer):
|
2018-01-02 21:22:12 +01:00
|
|
|
if len (sys.argv) == 1 or not os.path.exists (sys.argv[1]):
|
2018-03-15 21:04:31 +01:00
|
|
|
print ("""Failed to find hb-shape-fuzzer binary automatically,
|
2018-01-02 21:22:12 +01:00
|
|
|
please provide it as the first argument to the tool""")
|
|
|
|
sys.exit (1)
|
|
|
|
|
2018-03-15 21:04:31 +01:00
|
|
|
hb_shape_fuzzer = sys.argv[1]
|
2018-01-01 08:47:51 +01:00
|
|
|
|
2018-03-15 21:04:31 +01:00
|
|
|
print ('hb_shape_fuzzer:', hb_shape_fuzzer)
|
2017-12-29 20:43:29 +01:00
|
|
|
fails = 0
|
|
|
|
|
2018-10-20 06:09:18 +02:00
|
|
|
valgrind = None
|
|
|
|
if os.environ.get('RUN_VALGRIND', ''):
|
|
|
|
valgrind = which ('valgrind')
|
2019-04-15 22:52:21 +02:00
|
|
|
if valgrind is None:
|
|
|
|
print ("""Valgrind requested but not found.""")
|
|
|
|
sys.exit (1)
|
2018-10-19 07:54:21 +02:00
|
|
|
|
2018-10-17 01:39:29 +02:00
|
|
|
parent_path = os.path.join (srcdir, "fonts")
|
|
|
|
for file in os.listdir (parent_path):
|
2018-10-17 02:35:46 +02:00
|
|
|
path = os.path.join(parent_path, file)
|
2018-10-17 00:12:04 +02:00
|
|
|
|
2018-10-18 06:12:20 +02:00
|
|
|
text, returncode = cmd ([hb_shape_fuzzer, path])
|
2018-10-30 06:53:16 +01:00
|
|
|
if text.strip ():
|
|
|
|
print (text)
|
2017-12-29 20:43:29 +01:00
|
|
|
|
2018-10-19 07:54:21 +02:00
|
|
|
failed = False
|
2018-10-18 06:12:20 +02:00
|
|
|
if returncode != 0 or 'error' in text:
|
|
|
|
print ('failure on %s' % file)
|
2018-10-19 07:54:21 +02:00
|
|
|
failed = True
|
|
|
|
|
|
|
|
if valgrind:
|
2019-04-15 22:53:10 +02:00
|
|
|
text, returncode = cmd ([valgrind, '--error-exitcode=1', '--leak-check=full', hb_shape_fuzzer, path])
|
2018-10-19 07:54:21 +02:00
|
|
|
if returncode:
|
|
|
|
print (text)
|
|
|
|
print ('failure on %s' % file)
|
|
|
|
failed = True
|
|
|
|
|
|
|
|
if failed:
|
2017-12-29 20:43:29 +01:00
|
|
|
fails = fails + 1
|
|
|
|
|
|
|
|
if fails:
|
2018-03-15 23:41:08 +01:00
|
|
|
print ("%i shape fuzzer related tests failed." % fails)
|
2017-12-29 20:43:29 +01:00
|
|
|
sys.exit (1)
|