diff options
author | Angelos Mouzakitis <a.mouzakitis@virtualopensystems.com> | 2023-10-10 14:33:42 +0000 |
---|---|---|
committer | Angelos Mouzakitis <a.mouzakitis@virtualopensystems.com> | 2023-10-10 14:33:42 +0000 |
commit | af1a266670d040d2f4083ff309d732d648afba2a (patch) | |
tree | 2fc46203448ddcc6f81546d379abfaeb323575e9 /meson/mesonbuild/depfile.py | |
parent | e02cda008591317b1625707ff8e115a4841aa889 (diff) |
Change-Id: Iaf8d18082d3991dec7c0ebbea540f092188eb4ec
Diffstat (limited to 'meson/mesonbuild/depfile.py')
-rw-r--r-- | meson/mesonbuild/depfile.py | 85 |
1 files changed, 85 insertions, 0 deletions
diff --git a/meson/mesonbuild/depfile.py b/meson/mesonbuild/depfile.py new file mode 100644 index 000000000..62cbe8125 --- /dev/null +++ b/meson/mesonbuild/depfile.py @@ -0,0 +1,85 @@ +# Copyright 2019 Red Hat, Inc. +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at + +# http://www.apache.org/licenses/LICENSE-2.0 + +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +import collections + +def parse(lines): + rules = [] + targets = [] + deps = [] + in_deps = False + out = '' + for line in lines: + if not line.endswith('\n'): + line += '\n' + escape = None + for c in line: + if escape: + if escape == '$' and c != '$': + out += '$' + if escape == '\\' and c == '\n': + continue + out += c + escape = None + continue + if c == '\\' or c == '$': + escape = c + continue + elif c in (' ', '\n'): + if out != '': + if in_deps: + deps.append(out) + else: + targets.append(out) + out = '' + if c == '\n': + rules.append((targets, deps)) + targets = [] + deps = [] + in_deps = False + continue + elif c == ':': + targets.append(out) + out = '' + in_deps = True + continue + out += c + return rules + +Target = collections.namedtuple('Target', ['deps']) + +class DepFile: + def __init__(self, lines): + rules = parse(lines) + depfile = {} + for (targets, deps) in rules: + for target in targets: + t = depfile.setdefault(target, Target(deps=set())) + for dep in deps: + t.deps.add(dep) + self.depfile = depfile + + def get_all_dependencies(self, target, visited=None): + deps = set() + if not visited: + visited = set() + if target in visited: + return set() + visited.add(target) + target = self.depfile.get(target) + if not target: + return set() + deps.update(target.deps) + for dep in target.deps: + deps.update(self.get_all_dependencies(dep, visited)) + return sorted(deps) |