aboutsummaryrefslogtreecommitdiff
path: root/scripts/codeconverter/converter.py
blob: ebaf9b57ced5ab8bc51e87a7816d17c8b7bb9bc8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
#!/usr/bin/env python3
# QEMU library
#
# Copyright (C) 2020 Red Hat Inc.
#
# Authors:
#  Eduardo Habkost <ehabkost@redhat.com>
#
# This work is licensed under the terms of the GNU GPL, version 2.  See
# the COPYING file in the top-level directory.
#
import sys
import argparse
import os
import os.path
import re
from typing import *

from codeconverter.patching import FileInfo, match_class_dict, FileList
import codeconverter.qom_macros
from codeconverter.qom_type_info import TI_FIELDS, type_infos, TypeInfoVar

import logging
logger = logging.getLogger(__name__)
DBG = logger.debug
INFO = logger.info
WARN = logger.warning

def process_all_files(parser: argparse.ArgumentParser, args: argparse.Namespace) -> None:
    DBG("filenames: %r", args.filenames)

    files = FileList()
    files.extend(FileInfo(files, fn, args.force) for fn in args.filenames)
    for f in files:
        DBG('opening %s', f.filename)
        f.load()

    if args.table:
        fields = ['filename', 'variable_name'] + TI_FIELDS
        print('\t'.join(fields))
        for f in files:
            for t in f.matches_of_type(TypeInfoVar):
                assert isinstance(t, TypeInfoVar)
                values = [f.filename, t.name] + \
                         [t.get_initializer_value(f).raw
                          for f in TI_FIELDS]
                DBG('values: %r', values)
                assert all('\t' not in v for v in values)
                values = [v.replace('\n', ' ').replace('"', '') for v in values]
                print('\t'.join(values))
        return

    match_classes = match_class_dict()
    if not args.patterns:
        parser.error("--pattern is required")

    classes = [p for arg in args.patterns
                for p in re.split(r'[\s,]', arg)]
    for c in classes:
        if c not in match_classes:
            print("Invalid pattern name: %s" % (c), file=sys.stderr)
            print("Valid patterns:", file=sys.stderr)
            print(PATTERN_HELP, file=sys.stderr)
            sys.exit(1)

    DBG("classes: %r", classes)
    for f in files:
        DBG("patching contents of %s", f.filename)
        f.patch_content(max_passes=args.passes, class_names=classes)

    for f in files:
        #alltypes.extend(f.type_infos)
        #full_types.extend(f.full_types())

        if not args.dry_run:
            if args.inplace:
                f.patch_inplace()
            if args.diff:
                f.show_diff()
            if not args.diff and not args.inplace:
                f.write_to_file(sys.stdout)
                sys.stdout.flush()


PATTERN_HELP = ('\n'.join("  %s: %s" % (n, str(c.__doc__).strip())
                for (n,c) in sorted(match_class_dict().items())
                if c.has_replacement_rule()))

def main() -> None:
    p = argparse.ArgumentParser(formatter_class=argparse.RawDescriptionHelpFormatter)
    p.add_argument('filenames', nargs='+')
    p.add_argument('--passes', type=int, default=1,
                   help="Number of passes (0 means unlimited)")
    p.add_argument('--pattern', required=True, action='append',
                   default=[], dest='patterns',
                   help="Pattern to scan for")
    p.add_argument('--inplace', '-i', action='store_true',
                   help="Patch file in place")
    p.add_argument('--dry-run', action='store_true',
                   help="Don't patch files or print patching results")
    p.add_argument('--force', '-f', action='store_true',
                   help="Perform changes even if not completely safe")
    p.add_argument('--diff', action='store_true',
                   help="Print diff output on stdout")
    p.add_argument('--debug', '-d', action='store_true',
                   help="Enable debugging")
    p.add_argument('--verbose', '-v', action='store_true',
                   help="Verbose logging on stderr")
    p.add_argument('--table', action='store_true',
                   help="Print CSV table of type information")
    p.add_argument_group("Valid pattern names",
                         PATTERN_HELP)
    args = p.parse_args()

    loglevel = (logging.DEBUG if args.debug
             else logging.INFO if args.verbose
             else logging.WARN)
    logging.basicConfig(format='%(levelname)s: %(message)s', level=loglevel)
    DBG("args: %r", args)
    process_all_files(p, args)

if __name__ == '__main__':
    main()