| 
									
										
										
										
											2019-06-10 12:05:14 +02:00
										 |  |  | #! /bin/sh | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Python module for parsing and processing .ninja files. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Author: Paolo Bonzini | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Copyright (C) 2019 Red Hat, Inc. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # We don't want to put "#! @PYTHON@" as the shebang and | 
					
						
							|  |  |  | # make the file executable, so instead we make this a | 
					
						
							|  |  |  | # Python/shell polyglot.  The first line below starts a | 
					
						
							|  |  |  | # multiline string literal for Python, while it is just | 
					
						
							|  |  |  | # ":" for bash.  The closing of the multiline string literal | 
					
						
							|  |  |  | # is never parsed by bash since it exits before. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | '''':
 | 
					
						
							|  |  |  | case "$0" in | 
					
						
							|  |  |  |   /*) me=$0 ;; | 
					
						
							|  |  |  |   *) me=$(command -v "$0") ;; | 
					
						
							|  |  |  | esac | 
					
						
							|  |  |  | python="@PYTHON@" | 
					
						
							|  |  |  | case $python in | 
					
						
							|  |  |  |   @*) python=python3 ;; | 
					
						
							|  |  |  | esac | 
					
						
							|  |  |  | exec $python "$me" "$@" | 
					
						
							|  |  |  | exit 1 | 
					
						
							|  |  |  | '''
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from collections import namedtuple, defaultdict | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | import json | 
					
						
							|  |  |  | import argparse | 
					
						
							| 
									
										
										
										
											2020-09-01 07:13:16 -04:00
										 |  |  | import hashlib | 
					
						
							| 
									
										
										
										
											2019-06-10 12:05:14 +02:00
										 |  |  | import shutil | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class InvalidArgumentError(Exception): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # faster version of os.path.normpath: do nothing unless there is a double | 
					
						
							|  |  |  | # slash or a "." or ".." component.  The filter does not have to be super | 
					
						
							|  |  |  | # precise, but it has to be fast.  os.path.normpath is the hottest function | 
					
						
							|  |  |  | # for ninja2make without this optimization! | 
					
						
							|  |  |  | if os.path.sep == '/': | 
					
						
							|  |  |  |     def normpath(path, _slow_re=re.compile('/[./]')): | 
					
						
							|  |  |  |         return os.path.normpath(path) if _slow_re.search(path) or path[0] == '.' else path | 
					
						
							|  |  |  | else: | 
					
						
							|  |  |  |     normpath = os.path.normpath | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 07:13:16 -04:00
										 |  |  | def sha1_text(text): | 
					
						
							|  |  |  |     return hashlib.sha1(text.encode()).hexdigest() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-10 12:05:14 +02:00
										 |  |  | # ---- lexer and parser ---- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PATH_RE = r"[^$\s:|]+|\$[$ :]|\$[a-zA-Z0-9_-]+|\$\{[a-zA-Z0-9_.-]+\}" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 23:10:01 +08:00
										 |  |  | SIMPLE_PATH_RE = re.compile(r"^[^$\s:|]+$") | 
					
						
							| 
									
										
										
										
											2019-06-10 12:05:14 +02:00
										 |  |  | IDENT_RE = re.compile(r"[a-zA-Z0-9_.-]+$") | 
					
						
							|  |  |  | STRING_RE = re.compile(r"(" + PATH_RE + r"|[\s:|])(?:\r?\n)?|.") | 
					
						
							|  |  |  | TOPLEVEL_RE = re.compile(r"([=:#]|\|\|?|^ +|(?:" + PATH_RE + r")+)\s*|.") | 
					
						
							|  |  |  | VAR_RE=re.compile(r'\$\$|\$\{([^}]*)\}') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | BUILD = 1 | 
					
						
							|  |  |  | POOL = 2 | 
					
						
							|  |  |  | RULE = 3 | 
					
						
							|  |  |  | DEFAULT = 4 | 
					
						
							|  |  |  | EQUALS = 5 | 
					
						
							|  |  |  | COLON = 6 | 
					
						
							|  |  |  | PIPE = 7 | 
					
						
							|  |  |  | PIPE2 = 8 | 
					
						
							|  |  |  | IDENT = 9 | 
					
						
							|  |  |  | INCLUDE = 10 | 
					
						
							|  |  |  | INDENT = 11 | 
					
						
							|  |  |  | EOL = 12 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class LexerError(Exception): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ParseError(Exception): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class NinjaParserEvents(object): | 
					
						
							|  |  |  |     def __init__(self, parser): | 
					
						
							|  |  |  |         self.parser = parser | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def dollar_token(self, word, in_path=False): | 
					
						
							|  |  |  |         return '$$' if word == '$' else word | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def variable_expansion_token(self, varname): | 
					
						
							|  |  |  |         return '${%s}' % varname | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def variable(self, name, arg): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_file(self): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_file(self): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_scope(self): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_pool(self, name): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_rule(self, name): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_build(self, out, iout, rule, in_, iin, orderdep): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def default(self, targets): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class NinjaParser(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     InputFile = namedtuple('InputFile', 'filename iter lineno') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, filename, input): | 
					
						
							|  |  |  |         self.stack = [] | 
					
						
							|  |  |  |         self.top = None | 
					
						
							|  |  |  |         self.iter = None | 
					
						
							|  |  |  |         self.lineno = None | 
					
						
							|  |  |  |         self.match_keyword = False | 
					
						
							|  |  |  |         self.push(filename, input) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def file_changed(self): | 
					
						
							|  |  |  |         self.iter = self.top.iter | 
					
						
							|  |  |  |         self.lineno = self.top.lineno | 
					
						
							|  |  |  |         if self.top.filename is not None: | 
					
						
							|  |  |  |             os.chdir(os.path.dirname(self.top.filename) or '.') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def push(self, filename, input): | 
					
						
							|  |  |  |         if self.top: | 
					
						
							|  |  |  |             self.top.lineno = self.lineno | 
					
						
							|  |  |  |             self.top.iter = self.iter | 
					
						
							|  |  |  |             self.stack.append(self.top) | 
					
						
							|  |  |  |         self.top = self.InputFile(filename=filename or 'stdin', | 
					
						
							|  |  |  |                                   iter=self._tokens(input), lineno=0) | 
					
						
							|  |  |  |         self.file_changed() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def pop(self): | 
					
						
							|  |  |  |         if len(self.stack): | 
					
						
							|  |  |  |             self.top = self.stack[-1] | 
					
						
							|  |  |  |             self.stack.pop() | 
					
						
							|  |  |  |             self.file_changed() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.top = self.iter = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def next_line(self, input): | 
					
						
							|  |  |  |         line = next(input).rstrip() | 
					
						
							|  |  |  |         self.lineno += 1 | 
					
						
							|  |  |  |         while len(line) and line[-1] == '$': | 
					
						
							|  |  |  |             line = line[0:-1] + next(input).strip() | 
					
						
							|  |  |  |             self.lineno += 1 | 
					
						
							|  |  |  |         return line | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def print_token(self, tok): | 
					
						
							|  |  |  |         if tok == EOL: | 
					
						
							|  |  |  |             return "end of line" | 
					
						
							|  |  |  |         if tok == BUILD: | 
					
						
							|  |  |  |             return '"build"' | 
					
						
							|  |  |  |         if tok == POOL: | 
					
						
							|  |  |  |             return '"pool"' | 
					
						
							|  |  |  |         if tok == RULE: | 
					
						
							|  |  |  |             return '"rule"' | 
					
						
							|  |  |  |         if tok == DEFAULT: | 
					
						
							|  |  |  |             return '"default"' | 
					
						
							|  |  |  |         if tok == EQUALS: | 
					
						
							|  |  |  |             return '"="' | 
					
						
							|  |  |  |         if tok == COLON: | 
					
						
							|  |  |  |             return '":"' | 
					
						
							|  |  |  |         if tok == PIPE: | 
					
						
							|  |  |  |             return '"|"' | 
					
						
							|  |  |  |         if tok == PIPE2: | 
					
						
							|  |  |  |             return '"||"' | 
					
						
							|  |  |  |         if tok == INCLUDE: | 
					
						
							|  |  |  |             return '"include"' | 
					
						
							|  |  |  |         if tok == IDENT: | 
					
						
							|  |  |  |             return 'identifier' | 
					
						
							|  |  |  |         return '"%s"' % tok | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def error(self, msg): | 
					
						
							|  |  |  |         raise LexerError("%s:%d: %s" % (self.stack[-1].filename, self.lineno, msg)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def parse_error(self, msg): | 
					
						
							|  |  |  |         raise ParseError("%s:%d: %s" % (self.stack[-1].filename, self.lineno, msg)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def expected(self, expected, tok): | 
					
						
							|  |  |  |         msg = "found %s, expected " % (self.print_token(tok), ) | 
					
						
							|  |  |  |         for i, exp_tok in enumerate(expected): | 
					
						
							|  |  |  |             if i > 0: | 
					
						
							|  |  |  |                 msg = msg + (' or ' if i == len(expected) - 1 else ', ') | 
					
						
							|  |  |  |             msg = msg + self.print_token(exp_tok) | 
					
						
							|  |  |  |         self.parse_error(msg) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _variable_tokens(self, value): | 
					
						
							|  |  |  |         for m in STRING_RE.finditer(value): | 
					
						
							|  |  |  |             match = m.group(1) | 
					
						
							|  |  |  |             if not match: | 
					
						
							|  |  |  |                 self.error("unexpected '%s'" % (m.group(0), )) | 
					
						
							|  |  |  |             yield match | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _tokens(self, input): | 
					
						
							|  |  |  |         while True: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 line = self.next_line(input) | 
					
						
							|  |  |  |             except StopIteration: | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |             for m in TOPLEVEL_RE.finditer(line): | 
					
						
							|  |  |  |                 match = m.group(1) | 
					
						
							|  |  |  |                 if not match: | 
					
						
							|  |  |  |                     self.error("unexpected '%s'" % (m.group(0), )) | 
					
						
							|  |  |  |                 if match == ':': | 
					
						
							|  |  |  |                     yield COLON | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 if match == '|': | 
					
						
							|  |  |  |                     yield PIPE | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 if match == '||': | 
					
						
							|  |  |  |                     yield PIPE2 | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 if match[0] == ' ': | 
					
						
							|  |  |  |                     yield INDENT | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 if match[0] == '=': | 
					
						
							|  |  |  |                     yield EQUALS | 
					
						
							|  |  |  |                     value = line[m.start() + 1:].lstrip() | 
					
						
							|  |  |  |                     yield from self._variable_tokens(value) | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |                 if match[0] == '#': | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # identifier | 
					
						
							|  |  |  |                 if self.match_keyword: | 
					
						
							|  |  |  |                     if match == 'build': | 
					
						
							|  |  |  |                         yield BUILD | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     if match == 'pool': | 
					
						
							|  |  |  |                         yield POOL | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     if match == 'rule': | 
					
						
							|  |  |  |                         yield RULE | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     if match == 'default': | 
					
						
							|  |  |  |                         yield DEFAULT | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                     if match == 'include': | 
					
						
							|  |  |  |                         filename = line[m.start() + 8:].strip() | 
					
						
							|  |  |  |                         self.push(filename, open(filename, 'r')) | 
					
						
							|  |  |  |                         break | 
					
						
							|  |  |  |                     if match == 'subninja': | 
					
						
							|  |  |  |                         self.error('subninja is not supported') | 
					
						
							|  |  |  |                 yield match | 
					
						
							|  |  |  |             yield EOL | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def parse(self, events): | 
					
						
							|  |  |  |         global_var = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def look_for(*expected): | 
					
						
							|  |  |  |             # The last token in the token stream is always EOL.  This | 
					
						
							|  |  |  |             # is exploited to avoid catching StopIteration everywhere. | 
					
						
							|  |  |  |             tok = next(self.iter) | 
					
						
							|  |  |  |             if tok not in expected: | 
					
						
							|  |  |  |                 self.expected(expected, tok) | 
					
						
							|  |  |  |             return tok | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def look_for_ident(*expected): | 
					
						
							|  |  |  |             tok = next(self.iter) | 
					
						
							|  |  |  |             if isinstance(tok, str): | 
					
						
							|  |  |  |                 if not IDENT_RE.match(tok): | 
					
						
							|  |  |  |                     self.parse_error('variable expansion not allowed') | 
					
						
							|  |  |  |             elif tok not in expected: | 
					
						
							|  |  |  |                 self.expected(expected + (IDENT,), tok) | 
					
						
							|  |  |  |             return tok | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def parse_assignment_rhs(gen, expected, in_path): | 
					
						
							|  |  |  |             tokens = [] | 
					
						
							|  |  |  |             for tok in gen: | 
					
						
							|  |  |  |                 if not isinstance(tok, str): | 
					
						
							|  |  |  |                     if tok in expected: | 
					
						
							|  |  |  |                         break | 
					
						
							|  |  |  |                     self.expected(expected + (IDENT,), tok) | 
					
						
							|  |  |  |                 if tok[0] != '$': | 
					
						
							|  |  |  |                     tokens.append(tok) | 
					
						
							|  |  |  |                 elif tok == '$ ' or tok == '$$' or tok == '$:': | 
					
						
							|  |  |  |                     tokens.append(events.dollar_token(tok[1], in_path)) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     var = tok[2:-1] if tok[1] == '{' else tok[1:] | 
					
						
							|  |  |  |                     tokens.append(events.variable_expansion_token(var)) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 # gen must have raised StopIteration | 
					
						
							|  |  |  |                 tok = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if tokens: | 
					
						
							|  |  |  |                 # Fast path avoiding str.join() | 
					
						
							|  |  |  |                 value = tokens[0] if len(tokens) == 1 else ''.join(tokens) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 value = None | 
					
						
							|  |  |  |             return value, tok | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def look_for_path(*expected): | 
					
						
							|  |  |  |             # paths in build rules are parsed one space-separated token | 
					
						
							|  |  |  |             # at a time and expanded | 
					
						
							|  |  |  |             token = next(self.iter) | 
					
						
							|  |  |  |             if not isinstance(token, str): | 
					
						
							|  |  |  |                 return None, token | 
					
						
							|  |  |  |             # Fast path if there are no dollar and variable expansion | 
					
						
							|  |  |  |             if SIMPLE_PATH_RE.match(token): | 
					
						
							|  |  |  |                 return token, None | 
					
						
							|  |  |  |             gen = self._variable_tokens(token) | 
					
						
							|  |  |  |             return parse_assignment_rhs(gen, expected, True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def parse_assignment(tok): | 
					
						
							|  |  |  |             name = tok | 
					
						
							|  |  |  |             assert isinstance(name, str) | 
					
						
							|  |  |  |             look_for(EQUALS) | 
					
						
							|  |  |  |             value, tok = parse_assignment_rhs(self.iter, (EOL,), False) | 
					
						
							|  |  |  |             assert tok == EOL | 
					
						
							|  |  |  |             events.variable(name, value) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def parse_build(): | 
					
						
							|  |  |  |             # parse outputs | 
					
						
							|  |  |  |             out = [] | 
					
						
							|  |  |  |             iout = [] | 
					
						
							|  |  |  |             while True: | 
					
						
							|  |  |  |                 value, tok = look_for_path(COLON, PIPE) | 
					
						
							|  |  |  |                 if value is None: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |                 out.append(value) | 
					
						
							|  |  |  |             if tok == PIPE: | 
					
						
							|  |  |  |                 while True: | 
					
						
							|  |  |  |                     value, tok = look_for_path(COLON) | 
					
						
							|  |  |  |                     if value is None: | 
					
						
							|  |  |  |                         break | 
					
						
							|  |  |  |                     iout.append(value) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # parse rule | 
					
						
							|  |  |  |             assert tok == COLON | 
					
						
							|  |  |  |             rule = look_for_ident() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # parse inputs and dependencies | 
					
						
							|  |  |  |             in_ = [] | 
					
						
							|  |  |  |             iin = [] | 
					
						
							|  |  |  |             orderdep = [] | 
					
						
							|  |  |  |             while True: | 
					
						
							|  |  |  |                 value, tok = look_for_path(PIPE, PIPE2, EOL) | 
					
						
							|  |  |  |                 if value is None: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |                 in_.append(value) | 
					
						
							|  |  |  |             if tok == PIPE: | 
					
						
							|  |  |  |                 while True: | 
					
						
							|  |  |  |                     value, tok = look_for_path(PIPE2, EOL) | 
					
						
							|  |  |  |                     if value is None: | 
					
						
							|  |  |  |                         break | 
					
						
							|  |  |  |                     iin.append(value) | 
					
						
							|  |  |  |             if tok == PIPE2: | 
					
						
							|  |  |  |                 while True: | 
					
						
							|  |  |  |                     value, tok = look_for_path(EOL) | 
					
						
							|  |  |  |                     if value is None: | 
					
						
							|  |  |  |                         break | 
					
						
							|  |  |  |                     orderdep.append(value) | 
					
						
							|  |  |  |             assert tok == EOL | 
					
						
							|  |  |  |             events.begin_build(out, iout, rule, in_, iin, orderdep) | 
					
						
							|  |  |  |             nonlocal global_var | 
					
						
							|  |  |  |             global_var = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def parse_pool(): | 
					
						
							|  |  |  |             # pool declarations are ignored.  Just gobble all the variables | 
					
						
							|  |  |  |             ident = look_for_ident() | 
					
						
							|  |  |  |             look_for(EOL) | 
					
						
							|  |  |  |             events.begin_pool(ident) | 
					
						
							|  |  |  |             nonlocal global_var | 
					
						
							|  |  |  |             global_var = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def parse_rule(): | 
					
						
							|  |  |  |             ident = look_for_ident() | 
					
						
							|  |  |  |             look_for(EOL) | 
					
						
							|  |  |  |             events.begin_rule(ident) | 
					
						
							|  |  |  |             nonlocal global_var | 
					
						
							|  |  |  |             global_var = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def parse_default(): | 
					
						
							|  |  |  |             idents = [] | 
					
						
							|  |  |  |             while True: | 
					
						
							|  |  |  |                 ident = look_for_ident(EOL) | 
					
						
							|  |  |  |                 if ident == EOL: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |                 idents.append(ident) | 
					
						
							|  |  |  |             events.default(idents) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def parse_declaration(tok): | 
					
						
							|  |  |  |             if tok == EOL: | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             nonlocal global_var | 
					
						
							|  |  |  |             if tok == INDENT: | 
					
						
							|  |  |  |                 if global_var: | 
					
						
							|  |  |  |                     self.parse_error('indented line outside rule or edge') | 
					
						
							|  |  |  |                 tok = look_for_ident(EOL) | 
					
						
							|  |  |  |                 if tok == EOL: | 
					
						
							|  |  |  |                     return | 
					
						
							|  |  |  |                 parse_assignment(tok) | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if not global_var: | 
					
						
							|  |  |  |                 events.end_scope() | 
					
						
							|  |  |  |                 global_var = True | 
					
						
							|  |  |  |             if tok == POOL: | 
					
						
							|  |  |  |                 parse_pool() | 
					
						
							|  |  |  |             elif tok == BUILD: | 
					
						
							|  |  |  |                 parse_build() | 
					
						
							|  |  |  |             elif tok == RULE: | 
					
						
							|  |  |  |                 parse_rule() | 
					
						
							|  |  |  |             elif tok == DEFAULT: | 
					
						
							|  |  |  |                 parse_default() | 
					
						
							|  |  |  |             elif isinstance(tok, str): | 
					
						
							|  |  |  |                 parse_assignment(tok) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.expected((POOL, BUILD, RULE, INCLUDE, DEFAULT, IDENT), tok) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         events.begin_file() | 
					
						
							|  |  |  |         while self.iter: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 self.match_keyword = True | 
					
						
							|  |  |  |                 token = next(self.iter) | 
					
						
							|  |  |  |                 self.match_keyword = False | 
					
						
							|  |  |  |                 parse_declaration(token) | 
					
						
							|  |  |  |             except StopIteration: | 
					
						
							|  |  |  |                 self.pop() | 
					
						
							|  |  |  |         events.end_file() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # ---- variable handling ---- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def expand(x, rule_vars=None, build_vars=None, global_vars=None): | 
					
						
							|  |  |  |     if x is None: | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  |     changed = True | 
					
						
							|  |  |  |     have_dollar_replacement = False | 
					
						
							|  |  |  |     while changed: | 
					
						
							|  |  |  |         changed = False | 
					
						
							|  |  |  |         matches = list(VAR_RE.finditer(x)) | 
					
						
							|  |  |  |         if not matches: | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Reverse the match so that expanding later matches does not | 
					
						
							|  |  |  |         # invalidate m.start()/m.end() for earlier ones.  Do not reduce $$ to $ | 
					
						
							|  |  |  |         # until all variables are dealt with. | 
					
						
							|  |  |  |         for m in reversed(matches): | 
					
						
							|  |  |  |             name = m.group(1) | 
					
						
							|  |  |  |             if not name: | 
					
						
							|  |  |  |                 have_dollar_replacement = True | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             changed = True | 
					
						
							|  |  |  |             if build_vars and name in build_vars: | 
					
						
							|  |  |  |                 value = build_vars[name] | 
					
						
							|  |  |  |             elif rule_vars and name in rule_vars: | 
					
						
							|  |  |  |                 value = rule_vars[name] | 
					
						
							|  |  |  |             elif name in global_vars: | 
					
						
							|  |  |  |                 value = global_vars[name] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 value = '' | 
					
						
							|  |  |  |             x = x[:m.start()] + value + x[m.end():] | 
					
						
							|  |  |  |     return x.replace('$$', '$') if have_dollar_replacement else x | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Scope(object): | 
					
						
							|  |  |  |     def __init__(self, events): | 
					
						
							|  |  |  |         self.events = events | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def on_left_scope(self): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def on_variable(self, key, value): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class BuildScope(Scope): | 
					
						
							|  |  |  |     def __init__(self, events, out, iout, rule, in_, iin, orderdep, rule_vars): | 
					
						
							|  |  |  |         super().__init__(events) | 
					
						
							|  |  |  |         self.rule = rule | 
					
						
							|  |  |  |         self.out = [events.expand_and_normalize(x) for x in out] | 
					
						
							|  |  |  |         self.in_ = [events.expand_and_normalize(x) for x in in_] | 
					
						
							|  |  |  |         self.iin = [events.expand_and_normalize(x) for x in iin] | 
					
						
							|  |  |  |         self.orderdep = [events.expand_and_normalize(x) for x in orderdep] | 
					
						
							|  |  |  |         self.iout = [events.expand_and_normalize(x) for x in iout] | 
					
						
							|  |  |  |         self.rule_vars = rule_vars | 
					
						
							|  |  |  |         self.build_vars = dict() | 
					
						
							|  |  |  |         self._define_variable('out', ' '.join(self.out)) | 
					
						
							|  |  |  |         self._define_variable('in', ' '.join(self.in_)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def expand(self, x): | 
					
						
							|  |  |  |         return self.events.expand(x, self.rule_vars, self.build_vars) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def on_left_scope(self): | 
					
						
							|  |  |  |         self.events.variable('out', self.build_vars['out']) | 
					
						
							|  |  |  |         self.events.variable('in', self.build_vars['in']) | 
					
						
							|  |  |  |         self.events.end_build(self, self.out, self.iout, self.rule, self.in_, | 
					
						
							|  |  |  |                               self.iin, self.orderdep) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _define_variable(self, key, value): | 
					
						
							|  |  |  |         # The value has been expanded already, quote it for further | 
					
						
							|  |  |  |         # expansion from rule variables | 
					
						
							|  |  |  |         value = value.replace('$', '$$') | 
					
						
							|  |  |  |         self.build_vars[key] = value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def on_variable(self, key, value): | 
					
						
							|  |  |  |         # in and out are at the top of the lookup order and cannot | 
					
						
							|  |  |  |         # be overridden.  Also, unlike what the manual says, build | 
					
						
							|  |  |  |         # variables only lookup global variables.  They never lookup | 
					
						
							|  |  |  |         # rule variables, earlier build variables, or in/out. | 
					
						
							|  |  |  |         if key not in ('in', 'in_newline', 'out'): | 
					
						
							|  |  |  |             self._define_variable(key, self.events.expand(value)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class RuleScope(Scope): | 
					
						
							|  |  |  |     def __init__(self, events, name, vars_dict): | 
					
						
							|  |  |  |         super().__init__(events) | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.vars_dict = vars_dict | 
					
						
							|  |  |  |         self.generator = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def on_left_scope(self): | 
					
						
							|  |  |  |         self.events.end_rule(self, self.name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def on_variable(self, key, value): | 
					
						
							|  |  |  |         self.vars_dict[key] = value | 
					
						
							|  |  |  |         if key == 'generator': | 
					
						
							|  |  |  |             self.generator = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class NinjaParserEventsWithVars(NinjaParserEvents): | 
					
						
							|  |  |  |     def __init__(self, parser): | 
					
						
							|  |  |  |         super().__init__(parser) | 
					
						
							|  |  |  |         self.rule_vars = defaultdict(lambda: dict()) | 
					
						
							|  |  |  |         self.global_vars = dict() | 
					
						
							|  |  |  |         self.scope = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def variable(self, name, value): | 
					
						
							|  |  |  |         if self.scope: | 
					
						
							|  |  |  |             self.scope.on_variable(name, value) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.global_vars[name] = self.expand(value) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_build(self, out, iout, rule, in_, iin, orderdep): | 
					
						
							|  |  |  |         if rule != 'phony' and rule not in self.rule_vars: | 
					
						
							|  |  |  |             self.parser.parse_error("undefined rule '%s'" % rule) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.scope = BuildScope(self, out, iout, rule, in_, iin, orderdep, self.rule_vars[rule]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_pool(self, name): | 
					
						
							|  |  |  |         # pool declarations are ignored.  Just gobble all the variables | 
					
						
							|  |  |  |         self.scope = Scope(self) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_rule(self, name): | 
					
						
							|  |  |  |         if name in self.rule_vars: | 
					
						
							|  |  |  |             self.parser.parse_error("duplicate rule '%s'" % name) | 
					
						
							|  |  |  |         self.scope = RuleScope(self, name, self.rule_vars[name]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_scope(self): | 
					
						
							|  |  |  |         self.scope.on_left_scope() | 
					
						
							|  |  |  |         self.scope = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # utility functions: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def expand(self, x, rule_vars=None, build_vars=None): | 
					
						
							|  |  |  |         return expand(x, rule_vars, build_vars, self.global_vars) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def expand_and_normalize(self, x): | 
					
						
							|  |  |  |         return normpath(self.expand(x)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # extra events not present in the superclass: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_build(self, scope, out, iout, rule, in_, iin, orderdep): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_rule(self, scope, name): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # ---- test client that just prints back whatever it parsed  ---- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Writer(NinjaParserEvents): | 
					
						
							|  |  |  |     ARGS = argparse.ArgumentParser(description='Rewrite input build.ninja to stdout.') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, output, parser, args): | 
					
						
							|  |  |  |         super().__init__(parser) | 
					
						
							|  |  |  |         self.output = output | 
					
						
							|  |  |  |         self.indent = '' | 
					
						
							|  |  |  |         self.had_vars = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def dollar_token(self, word, in_path=False): | 
					
						
							|  |  |  |         return '$' + word | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def print(self, *args, **kwargs): | 
					
						
							|  |  |  |         if len(args): | 
					
						
							|  |  |  |             self.output.write(self.indent) | 
					
						
							|  |  |  |         print(*args, **kwargs, file=self.output) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def variable(self, name, value): | 
					
						
							|  |  |  |         self.print('%s = %s' % (name, value)) | 
					
						
							|  |  |  |         self.had_vars = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_scope(self): | 
					
						
							|  |  |  |         self.indent = '  ' | 
					
						
							|  |  |  |         self.had_vars = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_scope(self): | 
					
						
							|  |  |  |         if self.had_vars: | 
					
						
							|  |  |  |             self.print() | 
					
						
							|  |  |  |         self.indent = '' | 
					
						
							|  |  |  |         self.had_vars = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_pool(self, name): | 
					
						
							|  |  |  |         self.print('pool %s' % name) | 
					
						
							|  |  |  |         self.begin_scope() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_rule(self, name): | 
					
						
							|  |  |  |         self.print('rule %s' % name) | 
					
						
							|  |  |  |         self.begin_scope() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_build(self, outputs, implicit_outputs, rule, inputs, implicit, order_only): | 
					
						
							|  |  |  |         all_outputs = list(outputs) | 
					
						
							|  |  |  |         all_inputs = list(inputs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if implicit: | 
					
						
							|  |  |  |             all_inputs.append('|') | 
					
						
							|  |  |  |             all_inputs.extend(implicit) | 
					
						
							|  |  |  |         if order_only: | 
					
						
							|  |  |  |             all_inputs.append('||') | 
					
						
							|  |  |  |             all_inputs.extend(order_only) | 
					
						
							|  |  |  |         if implicit_outputs: | 
					
						
							|  |  |  |             all_outputs.append('|') | 
					
						
							|  |  |  |             all_outputs.extend(implicit_outputs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.print('build %s: %s' % (' '.join(all_outputs), | 
					
						
							|  |  |  |                                      ' '.join([rule] + all_inputs))) | 
					
						
							|  |  |  |         self.begin_scope() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def default(self, targets): | 
					
						
							|  |  |  |         self.print('default %s' % ' '.join(targets)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # ---- emit compile_commands.json ---- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Compdb(NinjaParserEventsWithVars): | 
					
						
							|  |  |  |     ARGS = argparse.ArgumentParser(description='Emit compile_commands.json.') | 
					
						
							|  |  |  |     ARGS.add_argument('rules', nargs='*', | 
					
						
							|  |  |  |                       help='The ninja rules to emit compilation commands for.') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, output, parser, args): | 
					
						
							|  |  |  |         super().__init__(parser) | 
					
						
							|  |  |  |         self.output = output | 
					
						
							|  |  |  |         self.rules = args.rules | 
					
						
							|  |  |  |         self.sep = '' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_file(self): | 
					
						
							|  |  |  |         self.output.write('[') | 
					
						
							|  |  |  |         self.directory = os.getcwd() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def print_entry(self, **entry): | 
					
						
							|  |  |  |         entry['directory'] = self.directory | 
					
						
							|  |  |  |         self.output.write(self.sep + json.dumps(entry)) | 
					
						
							|  |  |  |         self.sep = ',\n' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_build(self, out, iout, rule, in_, iin, orderdep): | 
					
						
							|  |  |  |         if in_ and rule in self.rules: | 
					
						
							|  |  |  |             super().begin_build(out, iout, rule, in_, iin, orderdep) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.scope = Scope(self) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_build(self, scope, out, iout, rule, in_, iin, orderdep): | 
					
						
							|  |  |  |         self.print_entry(command=scope.expand('${command}'), file=in_[0]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_file(self): | 
					
						
							|  |  |  |         self.output.write(']\n') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # ---- clean output files ---- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Clean(NinjaParserEventsWithVars): | 
					
						
							|  |  |  |     ARGS = argparse.ArgumentParser(description='Remove output build files.') | 
					
						
							|  |  |  |     ARGS.add_argument('-g', dest='generator', action='store_true', | 
					
						
							|  |  |  |                       help='clean generated files too') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, output, parser, args): | 
					
						
							|  |  |  |         super().__init__(parser) | 
					
						
							|  |  |  |         self.dry_run = args.dry_run | 
					
						
							|  |  |  |         self.verbose = args.verbose or args.dry_run | 
					
						
							|  |  |  |         self.generator = args.generator | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_file(self): | 
					
						
							|  |  |  |         print('Cleaning... ', end=(None if self.verbose else ''), flush=True) | 
					
						
							|  |  |  |         self.cnt = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_file(self): | 
					
						
							|  |  |  |         print('%d files' % self.cnt) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def do_clean(self, *files): | 
					
						
							|  |  |  |         for f in files: | 
					
						
							|  |  |  |             if self.dry_run: | 
					
						
							|  |  |  |                 if os.path.exists(f): | 
					
						
							|  |  |  |                     self.cnt += 1 | 
					
						
							|  |  |  |                     print('Would remove ' + f) | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     if os.path.isdir(f): | 
					
						
							|  |  |  |                         shutil.rmtree(f) | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         os.unlink(f) | 
					
						
							|  |  |  |                     self.cnt += 1 | 
					
						
							|  |  |  |                     if self.verbose: | 
					
						
							|  |  |  |                         print('Removed ' + f) | 
					
						
							|  |  |  |                 except FileNotFoundError: | 
					
						
							|  |  |  |                     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_build(self, scope, out, iout, rule, in_, iin, orderdep): | 
					
						
							|  |  |  |         if rule == 'phony': | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         if self.generator: | 
					
						
							|  |  |  |             rspfile = scope.expand('${rspfile}') | 
					
						
							|  |  |  |             if rspfile: | 
					
						
							|  |  |  |                 self.do_clean(rspfile) | 
					
						
							|  |  |  |         if self.generator or not scope.expand('${generator}'): | 
					
						
							|  |  |  |             self.do_clean(*out, *iout) | 
					
						
							|  |  |  |             depfile = scope.expand('${depfile}') | 
					
						
							|  |  |  |             if depfile: | 
					
						
							|  |  |  |                 self.do_clean(depfile) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # ---- convert build.ninja to makefile ---- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Ninja2Make(NinjaParserEventsWithVars): | 
					
						
							|  |  |  |     ARGS = argparse.ArgumentParser(description='Convert build.ninja to a Makefile.') | 
					
						
							|  |  |  |     ARGS.add_argument('--clean', dest='emit_clean', action='store_true', | 
					
						
							|  |  |  |                       help='Emit clean/distclean rules.') | 
					
						
							|  |  |  |     ARGS.add_argument('--doublecolon', action='store_true', | 
					
						
							|  |  |  |                       help='Emit double-colon rules for phony targets.') | 
					
						
							|  |  |  |     ARGS.add_argument('--omit', metavar='TARGET', nargs='+', | 
					
						
							|  |  |  |                       help='Targets to omit.') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, output, parser, args): | 
					
						
							|  |  |  |         super().__init__(parser) | 
					
						
							|  |  |  |         self.output = output | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.emit_clean = args.emit_clean | 
					
						
							|  |  |  |         self.doublecolon = args.doublecolon | 
					
						
							|  |  |  |         self.omit = set(args.omit) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if self.emit_clean: | 
					
						
							|  |  |  |             self.omit.update(['clean', 'distclean']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Lists of targets are kept in memory and emitted only at the | 
					
						
							|  |  |  |         # end because appending is really inefficient in GNU make. | 
					
						
							|  |  |  |         # We only do it when it's O(#rules) or O(#variables), but | 
					
						
							|  |  |  |         # never when it could be O(#targets). | 
					
						
							|  |  |  |         self.depfiles = list() | 
					
						
							|  |  |  |         self.rspfiles = list() | 
					
						
							|  |  |  |         self.build_vars = defaultdict(lambda: dict()) | 
					
						
							|  |  |  |         self.rule_targets = defaultdict(lambda: list()) | 
					
						
							|  |  |  |         self.stamp_targets = defaultdict(lambda: list()) | 
					
						
							|  |  |  |         self.all_outs = set() | 
					
						
							|  |  |  |         self.all_ins = set() | 
					
						
							|  |  |  |         self.all_phony = set() | 
					
						
							|  |  |  |         self.seen_default = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def print(self, *args, **kwargs): | 
					
						
							|  |  |  |         print(*args, **kwargs, file=self.output) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def dollar_token(self, word, in_path=False): | 
					
						
							|  |  |  |         if in_path and word == ' ': | 
					
						
							|  |  |  |             self.parser.parse_error('Make does not support spaces in filenames') | 
					
						
							|  |  |  |         return '$$' if word == '$' else word | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def print_phony(self, outs, ins): | 
					
						
							|  |  |  |         targets = ' '.join(outs).replace('$', '$$') | 
					
						
							|  |  |  |         deps = ' '.join(ins).replace('$', '$$') | 
					
						
							|  |  |  |         deps = deps.strip() | 
					
						
							|  |  |  |         if self.doublecolon: | 
					
						
							|  |  |  |             self.print(targets + '::' + (' ' if deps else '') + deps + ';@:') | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.print(targets + ':' + (' ' if deps else '') + deps) | 
					
						
							|  |  |  |         self.all_phony.update(outs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_file(self): | 
					
						
							|  |  |  |         self.print(r'# This is an automatically generated file, and it shows.') | 
					
						
							|  |  |  |         self.print(r'ninja-default:') | 
					
						
							|  |  |  |         self.print(r'.PHONY: ninja-default ninja-clean ninja-distclean') | 
					
						
							|  |  |  |         if self.emit_clean: | 
					
						
							|  |  |  |             self.print(r'ninja-clean:: ninja-clean-start; $(if $V,,@)rm -f ${ninja-depfiles}') | 
					
						
							|  |  |  |             self.print(r'ninja-clean-start:; $(if $V,,@echo Cleaning...)') | 
					
						
							|  |  |  |             self.print(r'ninja-distclean:: clean; $(if $V,,@)rm -f ${ninja-rspfiles}') | 
					
						
							|  |  |  |             self.print(r'.PHONY: ninja-clean-start') | 
					
						
							|  |  |  |             self.print_phony(['clean'], ['ninja-clean']) | 
					
						
							|  |  |  |             self.print_phony(['distclean'], ['ninja-distclean']) | 
					
						
							|  |  |  |         self.print(r'vpath') | 
					
						
							|  |  |  |         self.print(r'NULL :=') | 
					
						
							|  |  |  |         self.print(r'SPACE := ${NULL} #') | 
					
						
							|  |  |  |         self.print(r'MAKEFLAGS += -rR') | 
					
						
							|  |  |  |         self.print(r'define NEWLINE') | 
					
						
							|  |  |  |         self.print(r'') | 
					
						
							|  |  |  |         self.print(r'endef') | 
					
						
							|  |  |  |         self.print(r'.var.in_newline = $(subst $(SPACE),$(NEWLINE),${.var.in})') | 
					
						
							|  |  |  |         self.print(r"ninja-command = $(if $V,,$(if ${.var.description},@printf '%s\n' '$(subst ','\'',${.var.description})' && ))${.var.command}") | 
					
						
							|  |  |  |         self.print(r"ninja-command-restat = $(if $V,,$(if ${.var.description},@printf '%s\n' '$(subst ','\'',${.var.description})' && ))${.var.command} && if test -e $(firstword ${.var.out}); then printf '%s\n' ${.var.out} > $@; fi") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_file(self): | 
					
						
							|  |  |  |         def natural_sort_key(s, _nsre=re.compile('([0-9]+)')): | 
					
						
							|  |  |  |             return [int(text) if text.isdigit() else text.lower() | 
					
						
							|  |  |  |                     for text in _nsre.split(s)] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.print() | 
					
						
							|  |  |  |         self.print('ninja-outputdirs :=') | 
					
						
							|  |  |  |         for rule in self.rule_vars: | 
					
						
							|  |  |  |             if rule == 'phony': | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             self.print('ninja-targets-%s := %s' % (rule, ' '.join(self.rule_targets[rule]))) | 
					
						
							|  |  |  |             self.print('ninja-stamp-%s := %s' % (rule, ' '.join(self.stamp_targets[rule]))) | 
					
						
							|  |  |  |             self.print('ninja-outputdirs += $(sort $(dir ${ninja-targets-%s}))' % rule) | 
					
						
							|  |  |  |             self.print() | 
					
						
							|  |  |  |         self.print('dummy := $(shell mkdir -p . $(sort $(ninja-outputdirs)))') | 
					
						
							|  |  |  |         self.print('ninja-depfiles :=' + ' '.join(self.depfiles)) | 
					
						
							|  |  |  |         self.print('ninja-rspfiles :=' + ' '.join(self.rspfiles)) | 
					
						
							|  |  |  |         self.print('-include ${ninja-depfiles}') | 
					
						
							|  |  |  |         self.print() | 
					
						
							|  |  |  |         for targets in self.build_vars: | 
					
						
							|  |  |  |             for name, value in self.build_vars[targets].items(): | 
					
						
							| 
									
										
										
										
											2020-08-26 17:02:03 +02:00
										 |  |  |                 self.print('%s: private .var.%s := %s' % | 
					
						
							|  |  |  |                            (targets, name, value.replace('$', '$$'))) | 
					
						
							| 
									
										
										
										
											2019-06-10 12:05:14 +02:00
										 |  |  |             self.print() | 
					
						
							|  |  |  |         if not self.seen_default: | 
					
						
							|  |  |  |             default_targets = sorted(self.all_outs - self.all_ins, key=natural_sort_key) | 
					
						
							|  |  |  |             self.print('ninja-default: ' + ' '.join(default_targets)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # This is a hack...  Meson declares input meson.build files as | 
					
						
							|  |  |  |         # phony, because Ninja does not have an equivalent of Make's | 
					
						
							|  |  |  |         # "path/to/file:" declaration that ignores "path/to/file" even | 
					
						
							|  |  |  |         # if it is absent.  However, Makefile.ninja wants to depend on | 
					
						
							|  |  |  |         # build.ninja, which in turn depends on these phony targets which | 
					
						
							|  |  |  |         # would cause Makefile.ninja to be rebuilt in a loop. | 
					
						
							|  |  |  |         phony_targets = sorted(self.all_phony - self.all_ins, key=natural_sort_key) | 
					
						
							|  |  |  |         self.print('.PHONY: ' + ' '.join(phony_targets)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def variable(self, name, value): | 
					
						
							|  |  |  |         super().variable(name, value) | 
					
						
							|  |  |  |         if self.scope is None: | 
					
						
							|  |  |  |             self.global_vars[name] = self.expand(value) | 
					
						
							|  |  |  |             self.print('.var.%s := %s' % (name, self.global_vars[name])) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def begin_build(self, out, iout, rule, in_, iin, orderdep): | 
					
						
							|  |  |  |         if any(x in self.omit for x in out): | 
					
						
							|  |  |  |             self.scope = Scope(self) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         super().begin_build(out, iout, rule, in_, iin, orderdep) | 
					
						
							|  |  |  |         self.current_targets = ' '.join(self.scope.out + self.scope.iout).replace('$', '$$') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_build(self, scope, out, iout, rule, in_, iin, orderdep): | 
					
						
							|  |  |  |         self.rule_targets[rule] += self.scope.out | 
					
						
							|  |  |  |         self.rule_targets[rule] += self.scope.iout | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.all_outs.update(self.scope.iout) | 
					
						
							|  |  |  |         self.all_outs.update(self.scope.out) | 
					
						
							|  |  |  |         self.all_ins.update(self.scope.in_) | 
					
						
							|  |  |  |         self.all_ins.update(self.scope.iin) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         targets = self.current_targets | 
					
						
							|  |  |  |         self.current_targets = None | 
					
						
							|  |  |  |         if rule == 'phony': | 
					
						
							|  |  |  |             # Phony rules treat order-only dependencies as normal deps | 
					
						
							|  |  |  |             self.print_phony(out + iout, in_ + iin + orderdep) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         inputs = ' '.join(in_ + iin).replace('$', '$$') | 
					
						
							|  |  |  |         orderonly = ' '.join(orderdep).replace('$', '$$') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         rspfile = scope.expand('${rspfile}') | 
					
						
							|  |  |  |         if rspfile: | 
					
						
							|  |  |  |             rspfile_content = scope.expand('${rspfile_content}') | 
					
						
							|  |  |  |             with open(rspfile, 'w') as f: | 
					
						
							|  |  |  |                 f.write(rspfile_content) | 
					
						
							|  |  |  |             inputs += ' ' + rspfile | 
					
						
							|  |  |  |             self.rspfiles.append(rspfile) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         restat = 'restat' in self.scope.build_vars or 'restat' in self.rule_vars[rule] | 
					
						
							|  |  |  |         depfile = scope.expand('${depfile}') | 
					
						
							|  |  |  |         build_vars = { | 
					
						
							|  |  |  |             'command': scope.expand('${command}'), | 
					
						
							|  |  |  |             'description': scope.expand('${description}'), | 
					
						
							|  |  |  |             'out': scope.expand('${out}') | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if restat and not depfile: | 
					
						
							|  |  |  |             if len(out) == 1: | 
					
						
							|  |  |  |                 stamp = out[0] + '.stamp' | 
					
						
							|  |  |  |             else: | 
					
						
							| 
									
										
										
										
											2020-09-01 07:13:16 -04:00
										 |  |  |                 stamp = '%s@%s.stamp' % (rule, sha1_text(targets)[0:11]) | 
					
						
							| 
									
										
										
										
											2019-06-10 12:05:14 +02:00
										 |  |  |             self.print('%s: %s; @:' % (targets, stamp)) | 
					
						
							|  |  |  |             self.print('%s: %s | %s; ${ninja-command-restat}' % (stamp, inputs, orderonly)) | 
					
						
							|  |  |  |             self.rule_targets[rule].append(stamp) | 
					
						
							|  |  |  |             self.stamp_targets[rule].append(stamp) | 
					
						
							|  |  |  |             self.build_vars[stamp] = build_vars | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.print('%s: %s | %s; ${ninja-command}' % (targets, inputs, orderonly)) | 
					
						
							|  |  |  |             self.build_vars[targets] = build_vars | 
					
						
							|  |  |  |             if depfile: | 
					
						
							|  |  |  |                 self.depfiles.append(depfile) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def end_rule(self, scope, name): | 
					
						
							|  |  |  |         # Note that the generator pseudo-variable could also be attached | 
					
						
							|  |  |  |         # to a build block rather than a rule.  This is not handled here | 
					
						
							|  |  |  |         # in order to reduce the number of "rm" invocations.  However, | 
					
						
							|  |  |  |         # "ninjatool.py -t clean" does that correctly. | 
					
						
							|  |  |  |         target = 'distclean' if scope.generator else 'clean' | 
					
						
							|  |  |  |         self.print('ninja-%s:: ; $(if $V,,@)rm -f ${ninja-stamp-%s}' % (target, name)) | 
					
						
							|  |  |  |         if self.emit_clean: | 
					
						
							|  |  |  |             self.print('ninja-%s:: ; $(if $V,,@)rm -rf ${ninja-targets-%s}' % (target, name)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def default(self, targets): | 
					
						
							|  |  |  |         self.print("ninja-default: " + ' '.join(targets)) | 
					
						
							|  |  |  |         self.seen_default = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # ---- command line parsing ---- | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # we cannot use subparsers because tools are chosen through the "-t" | 
					
						
							|  |  |  | # option. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ToolAction(argparse.Action): | 
					
						
							|  |  |  |     def __init__(self, option_strings, dest, choices, metavar='TOOL', nargs=None, **kwargs): | 
					
						
							|  |  |  |         if nargs is not None: | 
					
						
							|  |  |  |             raise ValueError("nargs not allowed") | 
					
						
							|  |  |  |         super().__init__(option_strings, dest, required=True, choices=choices, | 
					
						
							|  |  |  |                          metavar=metavar, **kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __call__(self, parser, namespace, value, option_string): | 
					
						
							|  |  |  |         tool = self.choices[value] | 
					
						
							|  |  |  |         setattr(namespace, self.dest, tool) | 
					
						
							|  |  |  |         tool.ARGS.prog = '%s %s %s' % (parser.prog, option_string, value) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ToolHelpAction(argparse.Action): | 
					
						
							|  |  |  |     def __init__(self, option_strings, dest, nargs=None, **kwargs): | 
					
						
							|  |  |  |         if nargs is not None: | 
					
						
							|  |  |  |             raise ValueError("nargs not allowed") | 
					
						
							|  |  |  |         super().__init__(option_strings, dest, nargs=0, **kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __call__(self, parser, namespace, values, option_string=None): | 
					
						
							|  |  |  |         if namespace.tool: | 
					
						
							|  |  |  |             namespace.tool.ARGS.print_help() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             parser.print_help() | 
					
						
							|  |  |  |         parser.exit() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | tools = { | 
					
						
							|  |  |  |     'test': Writer, | 
					
						
							|  |  |  |     'ninja2make': Ninja2Make, | 
					
						
							|  |  |  |     'compdb': Compdb, | 
					
						
							|  |  |  |     'clean': Clean, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | parser = argparse.ArgumentParser(description='Process and transform build.ninja files.', | 
					
						
							|  |  |  |                                  add_help=False) | 
					
						
							|  |  |  | parser.add_argument('-C', metavar='DIR', dest='dir', default='.', | 
					
						
							|  |  |  |                     help='change to DIR before doing anything else') | 
					
						
							|  |  |  | parser.add_argument('-f', metavar='FILE', dest='file', default='build.ninja', | 
					
						
							|  |  |  |                     help='specify input build file [default=build.ninja]') | 
					
						
							|  |  |  | parser.add_argument('-n', dest='dry_run', action='store_true', | 
					
						
							|  |  |  |                     help='do not actually do anything') | 
					
						
							|  |  |  | parser.add_argument('-v', dest='verbose', action='store_true', | 
					
						
							|  |  |  |                     help='be more verbose') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | parser.add_argument('-t', dest='tool', choices=tools, action=ToolAction, | 
					
						
							|  |  |  |                     help='choose the tool to run') | 
					
						
							|  |  |  | parser.add_argument('-h', '--help', action=ToolHelpAction, | 
					
						
							|  |  |  |                     help='show this help message and exit') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if len(sys.argv) >= 2 and sys.argv[1] == '--version': | 
					
						
							|  |  |  |     print('1.8') | 
					
						
							|  |  |  |     sys.exit(0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | args, tool_args = parser.parse_known_args() | 
					
						
							|  |  |  | args.tool.ARGS.parse_args(tool_args, args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | os.chdir(args.dir) | 
					
						
							|  |  |  | with open(args.file, 'r') as f: | 
					
						
							|  |  |  |     parser = NinjaParser(args.file, f) | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         events = args.tool(sys.stdout, parser, args) | 
					
						
							|  |  |  |     except InvalidArgumentError as e: | 
					
						
							|  |  |  |         parser.error(str(e)) | 
					
						
							|  |  |  |     parser.parse(events) |