The Meson Build System
http://mesonbuild.com/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
129 lines
4.1 KiB
129 lines
4.1 KiB
# Copyright 2019 The Meson development team |
|
|
|
# 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. |
|
|
|
from .. import mesonlib |
|
|
|
def parse_generator_expressions(raw: str) -> str: |
|
'''Parse CMake generator expressions |
|
|
|
Most generator expressions are simply ignored for |
|
simplicety, however some are required for some common |
|
use cases. |
|
''' |
|
|
|
out = '' # type: str |
|
i = 0 # type: int |
|
|
|
def equal(arg: str) -> str: |
|
col_pos = arg.find(',') |
|
if col_pos < 0: |
|
return '0' |
|
else: |
|
return '1' if arg[:col_pos] == arg[col_pos + 1:] else '0' |
|
|
|
def vers_comp(op: str, arg: str) -> str: |
|
col_pos = arg.find(',') |
|
if col_pos < 0: |
|
return '0' |
|
else: |
|
return '1' if mesonlib.version_compare(arg[:col_pos], '{}{}'.format(op, arg[col_pos + 1:])) else '0' |
|
|
|
supported = { |
|
# Boolean functions |
|
'BOOL': lambda x: '0' if x.upper() in ['0', 'FALSE', 'OFF', 'N', 'NO', 'IGNORE', 'NOTFOUND'] or x.endswith('-NOTFOUND') else '1', |
|
'AND': lambda x: '1' if all([y == '1' for y in x.split(',')]) else '0', |
|
'OR': lambda x: '1' if any([y == '1' for y in x.split(',')]) else '0', |
|
'NOT': lambda x: '0' if x == '1' else '1', |
|
|
|
'0': lambda x: '', |
|
'1': lambda x: x, |
|
|
|
# String operations |
|
'STREQUAL': equal, |
|
'EQUAL': equal, |
|
'VERSION_LESS': lambda x: vers_comp('<', x), |
|
'VERSION_GREATER': lambda x: vers_comp('>', x), |
|
'VERSION_EQUAL': lambda x: vers_comp('=', x), |
|
'VERSION_LESS_EQUAL': lambda x: vers_comp('<=', x), |
|
'VERSION_GREATER_EQUAL': lambda x: vers_comp('>=', x), |
|
|
|
# String modification |
|
'LOWER_CASE': lambda x: x.lower(), |
|
'UPPER_CASE': lambda x: x.upper(), |
|
|
|
# Always assume the BUILD_INTERFACE is valid. |
|
# INSTALL_INTERFACE is always invalid for subprojects and |
|
# it should also never appear in CMake config files, used |
|
# for dependencies |
|
'INSTALL_INTERFACE': lambda x: '', |
|
'BUILD_INTERFACE': lambda x: x, |
|
|
|
# Constants |
|
'ANGLE-R': lambda x: '>', |
|
'COMMA': lambda x: ',', |
|
'SEMICOLON': lambda x: ';', |
|
} |
|
|
|
# Recursively evaluate generator expressions |
|
def eval_generator_expressions() -> str: |
|
nonlocal i |
|
i += 2 |
|
|
|
func = '' # type: str |
|
args = '' # type: str |
|
res = '' # type: str |
|
exp = '' # type: str |
|
|
|
# Determine the body of the expression |
|
while i < len(raw): |
|
if raw[i] == '>': |
|
# End of the generator expression |
|
break |
|
elif i < len(raw) - 1 and raw[i] == '$' and raw[i + 1] == '<': |
|
# Nested generator expression |
|
exp += eval_generator_expressions() |
|
else: |
|
# Generator expression body |
|
exp += raw[i] |
|
|
|
i += 1 |
|
|
|
# Split the expression into a function and arguments part |
|
col_pos = exp.find(':') |
|
if col_pos < 0: |
|
func = exp |
|
else: |
|
func = exp[:col_pos] |
|
args = exp[col_pos + 1:] |
|
|
|
func = func.strip() |
|
args = args.strip() |
|
|
|
# Evaluate the function |
|
if func in supported: |
|
res = supported[func](args) |
|
|
|
return res |
|
|
|
while i < len(raw): |
|
if i < len(raw) - 1 and raw[i] == '$' and raw[i + 1] == '<': |
|
# Generator expression detected --> try resolving it |
|
out += eval_generator_expressions() |
|
else: |
|
# Normal string, leave unchanged |
|
out += raw[i] |
|
|
|
i += 1 |
|
|
|
return out
|
|
|