| #! /usr/bin/env python |
| # Copyright 2018 Google LLC. |
| # Use of this source code is governed by a BSD-style license that can be found in the LICENSE file. |
| |
| import json |
| import md5 |
| import multiprocessing |
| import os |
| import shutil |
| import sys |
| import tempfile |
| import urllib |
| import urllib2 |
| |
| from subprocess import check_call, check_output |
| |
| assert '/' in [os.sep, os.altsep] and os.pardir == '..' |
| |
| ASSETS = 'platform_tools/android/apps/skqp/src/main/assets' |
| BUCKET = 'skia-skqp-assets' |
| |
| def urlopen(url): |
| cookie = os.environ.get('SKIA_GOLD_COOKIE', '') |
| return urllib2.urlopen(urllib2.Request(url, headers={'Cookie': cookie})) |
| |
| def make_skqp_model(arg): |
| name, urls, exe = arg |
| tmp = tempfile.mkdtemp() |
| for url in urls: |
| urllib.urlretrieve(url, tmp + '/' + url[url.rindex('/') + 1:]) |
| check_call([exe, tmp, ASSETS + '/gmkb/' + name]) |
| shutil.rmtree(tmp) |
| sys.stdout.write(name + ' ') |
| sys.stdout.flush() |
| |
| def goldgetter(meta, exe): |
| assert os.path.exists(exe) |
| jobs = [] |
| for rec in meta: |
| urls = [d['URL'] for d in rec['digests'] |
| if d['status'] == 'positive' and |
| (set(d['paramset']['config']) & set(['vk', 'gles']))] |
| if urls: |
| jobs.append((rec['testName'], urls, exe)) |
| pool = multiprocessing.Pool(processes=20) |
| pool.map(make_skqp_model, jobs) |
| sys.stdout.write('\n') |
| return set((n for n, _, _ in jobs)) |
| |
| def gold(first_commit, last_commit): |
| c1, c2 = (check_output(['git', 'rev-parse', c]).strip() |
| for c in (first_commit, last_commit)) |
| f = urlopen('https://public-gold.skia.org/json/export?' + urllib.urlencode([ |
| ('fbegin', c1), |
| ('fend', c2), |
| ('query', 'config=gles&config=vk&source_type=gm'), |
| ('pos', 'true'), |
| ('neg', 'false'), |
| ('unt', 'false') |
| ])) |
| j = json.load(f) |
| f.close() |
| return j |
| |
| def gset(path): |
| s = set() |
| if os.path.isfile(path): |
| with open(path, 'r') as f: |
| for line in f: |
| s.add(line.strip()) |
| return s |
| |
| def make_rendertest_list(models, good, bad): |
| assert good.isdisjoint(bad) |
| do_score = good & models |
| no_score = bad | (good - models) |
| to_delete = models & bad |
| for d in to_delete: |
| path = ASSETS + '/gmkb/' + d |
| if os.path.isdir(path): |
| shutil.rmtree(path) |
| results = dict() |
| for n in do_score: |
| results[n] = 0 |
| for n in no_score: |
| results[n] = -1 |
| return ''.join('%s,%d\n' % (n, results[n]) for n in sorted(results)) |
| |
| def get_digest(path): |
| m = md5.new() |
| with open(path, 'r') as f: |
| m.update(f.read()) |
| return m.hexdigest() |
| |
| def upload_cmd(path, digest): |
| return ['gsutil', 'cp', path, 'gs://%s/%s' % (BUCKET, digest)] |
| |
| def upload_model(): |
| bucket_url = 'gs://%s/' % BUCKET |
| extant = set((u.replace(bucket_url, '', 1) |
| for u in check_output(['gsutil', 'ls', bucket_url]).splitlines() if u)) |
| cmds = [] |
| filelist = [] |
| for dirpath, _, filenames in os.walk(ASSETS + '/gmkb'): |
| for filename in filenames: |
| path = os.path.join(dirpath, filename) |
| digest = get_digest(path) |
| if digest not in extant: |
| cmds.append(upload_cmd(path, digest)) |
| filelist.append('%s;%s\n' % (digest, os.path.relpath(path, ASSETS))) |
| tmp = tempfile.mkdtemp() |
| filelist_path = tmp + '/x' |
| with open(filelist_path, 'w') as o: |
| for l in filelist: |
| o.write(l) |
| filelist_digest = get_digest(filelist_path) |
| if filelist_digest not in extant: |
| cmds.append(upload_cmd(filelist_path, filelist_digest)) |
| |
| pool = multiprocessing.Pool(processes=20) |
| pool.map(check_call, cmds) |
| shutil.rmtree(tmp) |
| return filelist_digest |
| |
| def remove(x): |
| if os.path.isdir(x) and not os.path.islink(x): |
| shutil.rmtree(x) |
| if os.path.exists(x): |
| os.remove(x) |
| |
| def main(first_commit, last_commit): |
| check_call(upload_cmd('/dev/null', get_digest('/dev/null'))) |
| |
| os.chdir(os.path.dirname(__file__) + '/../..') |
| remove(ASSETS + '/files.checksum') |
| for d in [ASSETS + '/gmkb', ASSETS + '/skqp', ]: |
| remove(d) |
| os.mkdir(d) |
| |
| check_call([sys.executable, 'tools/git-sync-deps'], |
| env=dict(os.environ, GIT_SYNC_DEPS_QUIET='T')) |
| build = 'out/ndebug' |
| check_call(['bin/gn', 'gen', build, |
| '--args=cc="clang" cxx="clang++" is_debug=false']) |
| check_call(['ninja', '-C', build, |
| 'jitter_gms', 'list_gpu_unit_tests', 'make_skqp_model']) |
| |
| models = goldgetter(gold(first_commit, last_commit), build + '/make_skqp_model') |
| |
| check_call([build + '/jitter_gms', 'tools/skqp/bad_gms.txt']) |
| |
| with open(ASSETS + '/skqp/rendertests.txt', 'w') as o: |
| o.write(make_rendertest_list(models, gset('good.txt'), gset('bad.txt'))) |
| |
| remove('good.txt') |
| remove('bad.txt') |
| |
| with open(ASSETS + '/skqp/unittests.txt', 'w') as o: |
| o.write(check_output([build + '/list_gpu_unit_tests'])) |
| |
| with open(ASSETS + '/files.checksum', 'w') as o: |
| o.write(upload_model() + '\n') |
| |
| sys.stdout.write(ASSETS + '/files.checksum\n') |
| sys.stdout.write(ASSETS + '/skqp/rendertests.txt\n') |
| sys.stdout.write(ASSETS + '/skqp/unittests.txt\n') |
| |
| if __name__ == '__main__': |
| if len(sys.argv) != 3: |
| sys.stderr.write('Usage:\n %s C1 C2\n\n' % sys.argv[0]) |
| sys.exit(1) |
| main(sys.argv[1], sys.argv[2]) |