style.py revision 11409:72f80dd8b194
1#! /usr/bin/env python
2#
3# Copyright (c) 2016 ARM Limited
4# All rights reserved
5#
6# The license below extends only to copyright in the software and shall
7# not be construed as granting a license to any other intellectual
8# property including but not limited to intellectual property relating
9# to a hardware implementation of the functionality of the software
10# licensed hereunder.  You may use the software subject to the license
11# terms below provided that you ensure that this notice is replicated
12# unmodified and in its entirety in all distributions of the software,
13# modified or unmodified, in source code or in binary form.
14#
15# Redistribution and use in source and binary forms, with or without
16# modification, are permitted provided that the following conditions are
17# met: redistributions of source code must retain the above copyright
18# notice, this list of conditions and the following disclaimer;
19# redistributions in binary form must reproduce the above copyright
20# notice, this list of conditions and the following disclaimer in the
21# documentation and/or other materials provided with the distribution;
22# neither the name of the copyright holders nor the names of its
23# contributors may be used to endorse or promote products derived from
24# this software without specific prior written permission.
25#
26# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
27# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
28# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
29# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
30# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
31# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
32# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
33# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
34# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
35# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
36# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
37#
38# Authors: Andreas Sandberg
39
40import os
41import sys
42
43from style.file_types import lang_type
44from style.verifiers import all_verifiers
45from style.region import all_regions
46
47from style.style import StdioUI
48from style import repo
49
50def verify(filename, regions=all_regions, verbose=False):
51    ui = StdioUI()
52    opts = {}
53    base = os.path.join(os.path.dirname(__file__), "..")
54    verifiers = [ v(ui, opts, base=base) for v in all_verifiers ]
55
56    if verbose:
57        print "Verifying %s[%s]..." % (filename, regions)
58    for verifier in verifiers:
59        if verbose:
60            print "Applying %s (%s)" % (
61                verifier.test_name, verifier.__class__.__name__)
62        if verifier.apply(filename, regions=regions):
63            return False
64    return True
65
66def detect_repo():
67    repo_classes = repo.detect_repo()
68    if not repo_classes:
69        print >> sys.stderr, "Error: Failed to detect repository type, no " \
70            "known repository type found."
71        sys.exit(1)
72    elif len(repo_classes) > 1:
73        print >> sys.stderr, "Error: Detected multiple repository types."
74        sys.exit(1)
75    else:
76        return repo_classes[0]()
77
78repo_types = {
79    "auto" : detect_repo,
80    "none" : lambda : None,
81    "git" : repo.GitRepo,
82    "hg" : repo.MercurialRepo,
83}
84
85if __name__ == '__main__':
86    import argparse
87
88    parser = argparse.ArgumentParser(
89        description="Check a file for gem5 style violations",
90        epilog="""If no files are specified, the style checker tries to
91        determine the list of modified and added files from the version
92        control system and checks those."""
93    )
94
95    parser.add_argument("--verbose", "-v", action="count",
96                        help="Produce verbose output")
97
98    parser.add_argument("--modifications", "-m", action="store_true",
99                        help="""Apply the style checker to modified regions
100                        instead of whole files""")
101
102    parser.add_argument("--repo-type", choices=repo_types, default="auto",
103                        help="Repository type to use to detect changes")
104
105    parser.add_argument("files", metavar="FILE", nargs="*",
106                        type=str,
107                        help="Source file(s) to inspect")
108
109    args = parser.parse_args()
110
111    repo = repo_types[args.repo_type]()
112
113    files = args.files
114    if not files and repo:
115        added, modified = repo.staged_files()
116        files = [ repo.file_path(f) for f in added + modified ]
117
118    for filename in files:
119        if args.modifications and repo and repo.in_repo(filename):
120            regions = repo.modified_regions(filename)
121        else:
122            regions = all_regions
123
124        if not verify(filename, regions=regions,
125                      verbose=args.verbose):
126            sys.exit(1)
127