summaryrefslogtreecommitdiffstats
path: root/python/makevars.py
blob: 951cd3438b75fe05c635998613a406b0d6979144 (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
#
# helper class to grab variables from FRR's Makefile
#

import os
import subprocess
import re


class MakeVarsBase(object):
    """
    common code between MakeVars and MakeReVars
    """

    def __init__(self):
        self._data = dict()

    def __getitem__(self, k):
        if k not in self._data:
            self.getvars([k])
        return self._data[k]

    def get(self, k, defval=None):
        if k not in self._data:
            self.getvars([k])
        return self._data.get(k) or defval


class MakeVars(MakeVarsBase):
    """
    makevars['FOO_CFLAGS'] gets you "FOO_CFLAGS" from Makefile

    This variant works by invoking make as a subprocess, i.e. Makefile must
    be valid and working.  (This is sometimes a problem if depfiles have not
    been generated.)
    """

    def getvars(self, varlist):
        """
        get a batch list of variables from make.  faster than individual calls.
        """
        rdfd, wrfd = os.pipe()

        shvars = ["shvar-%s" % s for s in varlist]
        make = subprocess.Popen(
            ["make", "-s", "VARFD=%d" % wrfd] + shvars, pass_fds=[wrfd]
        )
        os.close(wrfd)
        data = b""

        rdf = os.fdopen(rdfd, "rb")
        while True:
            rdata = rdf.read()
            if len(rdata) == 0:
                break
            data += rdata

        del rdf
        make.wait()

        data = data.decode("US-ASCII").strip().split("\n")
        for row in data:
            k, v = row.split("=", 1)
            v = v[1:-1]
            self._data[k] = v


class MakeReVars(MakeVarsBase):
    """
    makevars['FOO_CFLAGS'] gets you "FOO_CFLAGS" from Makefile

    This variant works by regexing through Makefile.  This means the Makefile
    does not need to be fully working, but on the other hand it doesn't support
    fancy complicated make expressions.
    """

    var_re = re.compile(
        r"^([^=#\n\s]+)[ \t]*=[ \t]*([^#\n]*)(?:#.*)?$", flags=re.MULTILINE
    )
    repl_re = re.compile(r"\$(?:([A-Za-z])|\(([^\)]+)\))")

    def __init__(self, maketext):
        super(MakeReVars, self).__init__()
        self._vars = dict(self.var_re.findall(maketext.replace("\\\n", "")))

    def replacevar(self, match):
        varname = match.group(1) or match.group(2)
        return self._vars.get(varname, "")

    def getvars(self, varlist):
        for varname in varlist:
            if varname not in self._vars:
                continue

            val, prevval = self._vars[varname], None
            while val != prevval:
                prevval = val
                val = self.repl_re.sub(self.replacevar, val)

            self._data[varname] = val