compile_all.py 6.7 KB
Newer Older
Enrico Pozzobon committed
1 2 3 4
#!/usr/bin/env python3

import os
import sys
Enrico Pozzobon committed
5
import stat
6
import argparse
Enrico Pozzobon committed
7 8 9
import shutil
import subprocess

Enrico Pozzobon committed
10

Enrico Pozzobon committed
11
def build(algo_dir, template_dir, build_dir):
12 13 14
    if os.path.isdir(build_dir):
        return None

15
    print("Building in %s" % build_dir)
16

Enrico Pozzobon committed
17
    # copy all the files from the submitted algorithm into the build directory
Enrico Pozzobon committed
18 19
    shutil.copytree(algo_dir, build_dir)

Enrico Pozzobon committed
20
    # remove the test vectors generator if it is there
Enrico Pozzobon committed
21 22 23 24
    c = os.path.join(build_dir, "genkat_aead.c")
    if os.path.exists(c):
        os.remove(c)

Enrico Pozzobon committed
25
    # find all c and h files, since they will be added to the makefile
26 27
    hfiles = []
    cfiles = []
28
    sfiles = []
29 30
    for r, d, f in os.walk(build_dir):
        for file in f:
31
            if file.lower().endswith(".c"):
32
                cfiles.append(file)
33 34 35
            elif file.lower().endswith(".s"):
                sfiles.append(file)
            elif file.lower().endswith(".h"):
36 37
                hfiles.append(file)

Enrico Pozzobon committed
38
    # copy all the files from the template directory into the build directory
39 40 41 42 43 44 45 46 47 48
    for f in os.listdir(template_dir):
        dst = os.path.join(build_dir, f)
        src = os.path.join(template_dir, f)
        if os.path.isfile(src) or os.path.islink(src):
            shutil.copy2(src, dst)
        elif os.path.isdir(src):
            shutil.copytree(src, dst)
        else:
            raise Exception("I don't know what %s is" % src)

Enrico Pozzobon committed
49
    # prepare the environmental variables for the makefile
Enrico Pozzobon committed
50 51
    env = os.environ
    env['SRC_FILES'] = ' '.join(cfiles)
52
    env['ASM_FILES'] = ' '.join(sfiles)
Enrico Pozzobon committed
53 54
    env['HDR_FILES'] = ' '.join(hfiles)

Enrico Pozzobon committed
55
    # enter the directory and execute the makefile
Enrico Pozzobon committed
56
    wd = os.getcwd()
Enrico Pozzobon committed
57
    os.chdir(build_dir)
Enrico Pozzobon committed
58
    try:
59
        if os.path.isfile('./configure'):
60
            subprocess.check_call(["./configure"])
61

62 63 64 65 66
        stdout_path = 'make.stdout.log'
        stderr_path = 'make.stderr.log'
        with open(stdout_path, 'w') as outfile, \
             open(stderr_path, 'w') as errfile:
            subprocess.check_call(['make'], stdout=outfile, stderr=errfile)
Enrico Pozzobon committed
67

68
        if os.path.isfile('./cleanup'):
69
            subprocess.check_call(["./cleanup"])
70

Enrico Pozzobon committed
71
    finally:
72 73
        sys.stdout.flush()
        sys.stderr.flush()
Enrico Pozzobon committed
74 75
        os.chdir(wd)

Enrico Pozzobon committed
76
    # if execution arrives here, the build was successful
Enrico Pozzobon committed
77 78
    return build_dir

Enrico Pozzobon committed
79 80

# Find test vectors in directory or one of the parent directories
Enrico Pozzobon committed
81 82
def find_test_vectors(d):
    kat = None
Enrico Pozzobon committed
83
    while True:
Enrico Pozzobon committed
84 85 86 87 88 89 90
        if d == '':
            raise Exception("Test vector not found")
        for f in os.listdir(d):
            if f.startswith("LWC_AEAD_KAT_") and f.endswith(".txt"):
                if kat is not None:
                    raise Exception("Multiple test vectors?")
                kat = f
Enrico Pozzobon committed
91 92 93 94
        if kat is None:
            d = os.path.split(d)[0]
        else:
            break
Enrico Pozzobon committed
95 96
    kat = os.path.join(d, kat)
    return kat
Enrico Pozzobon committed
97

Enrico Pozzobon committed
98

99
def main(argv):
100
    include_list = None
101 102 103

    # Parse the arguments
    argparser = argparse.ArgumentParser(
104
        description='Compiles all LWC submissions for a given template')
105 106 107

    argparser.add_argument('-v', '--verbose', action='count')
    argparser.add_argument('-i', '--include', action='append')
108 109 110 111
    argparser.add_argument('-t', '--template', default='templates/linux')
    argparser.add_argument('-b', '--build-dir', default='build')
    argparser.add_argument('-s', '--submissions-dir',
                           default='all-lwc-submission-files')
112 113 114 115 116 117

    args = argparser.parse_args(argv[1:])
    template_dir = args.template
    build_root_dir = args.build_dir

    include_list = args.include
118
    submissions_dir = args.submissions_dir
119

120
    print("Using template %s" % template_dir)
Enrico Pozzobon committed
121 122
    subs = os.listdir(submissions_dir)

Enrico Pozzobon committed
123
    # get all the submissions by looking for files named "api.h"
124
    implementations = []
Enrico Pozzobon committed
125
    for submission in subs:
126
        variants_dir = os.path.join(
127
            submissions_dir, submission, "Implementations", "crypto_aead")
Enrico Pozzobon committed
128

129
        if not os.path.isdir(variants_dir):
Enrico Pozzobon committed
130 131
            continue

132
        if "NOT ACCEPTED" in variants_dir:
Enrico Pozzobon committed
133 134 135 136 137 138
            continue

        print()
        print("###  %s  ###" % submission)

        c = 0
139 140 141 142 143 144 145 146 147
        for variant in os.listdir(variants_dir):
            implementations_dir = os.path.join(
                variants_dir, variant)
            for implementation in os.listdir(implementations_dir):
                implementation_dir = os.path.join(
                    implementations_dir, implementation)
                if os.path.isfile(os.path.join(implementation_dir, "api.h")):
                    implementations.append(
                        (submission, variant, implementation))
148 149 150 151
                    c += 1

        if c == 0:
            raise Exception("No implementations found")
Enrico Pozzobon committed
152

153 154
    if include_list is not None:
        print("Include list has %d entries" % len(include_list))
Enrico Pozzobon committed
155

156
    files = []
157 158 159 160 161
    for submission, variant, implementation in implementations:

        # base name n (a.k.a. cipher slug)
        n = '.'.join([submission, variant, implementation])
        print(n)
Enrico Pozzobon committed
162

163
        # Source directory d
164 165 166
        d = os.path.join(
            submissions_dir, submission, "Implementations", "crypto_aead",
            variant, implementation)
167 168
        assert os.path.isdir(d)
        print(d)
Enrico Pozzobon committed
169

170 171 172
        # Test vectors file t
        t = find_test_vectors(d)
        print(t)
Enrico Pozzobon committed
173

174 175
        # if include_list was provided, skip elements not in the list
        if include_list is not None:
176
            if n not in include_list:
177 178
                continue

179 180 181 182 183 184 185
        # Find date of last modification in directory
        st_mtime = 0
        for root, dirs, filess in os.walk(d):
            for name in filess:
                path = os.path.join(root, name)
                st_mtime = max(st_mtime, os.stat(path).st_mtime)

186
        # Put all in a tuple and count
187
        files.append((t, d, n, st_mtime))
Enrico Pozzobon committed
188

189
    # Uncomment next line for testing, if we only want to do 1
190
    # files = files[:1]
191
    print("%d algorithms will be compiled" % len(files))
Enrico Pozzobon committed
192

193 194
    if not os.path.isdir(build_root_dir):
        os.mkdir(build_root_dir)
Enrico Pozzobon committed
195 196 197

    print()

198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
    # Build all found algorithms
    for i, (t, d, name, st_mtime) in enumerate(files):
        print()
        print(d)
        try:
            build_dir = os.path.join(build_root_dir, name)

            b = build(d, template_dir, build_dir)

            if b is None:
                continue

            shutil.copyfile(t, os.path.join(b, 'LWC_AEAD_KAT.txt'))

            mdate_path = os.path.join(build_dir, 'cipher_mtime.txt')
            with open(mdate_path, 'wt') as mdate_file:
                print(int(st_mtime), file=mdate_file)

            print("COMPILATION SUCCESS FOR %s" % d)
        except Exception as ex:
            print("COMPILATION FAILED FOR %s" % d)
            print(ex)
Enrico Pozzobon committed
220

221 222
    print()

Enrico Pozzobon committed
223 224 225

if __name__ == "__main__":
    sys.exit(main(sys.argv))