• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1#!/usr/bin/python3 -i
2#
3# Copyright 2013-2024 The Khronos Group Inc.
4#
5# SPDX-License-Identifier: Apache-2.0
6"""Utilities for working with attributes of the XML registry."""
7
8import re
9
10_PARAM_REF_NAME_RE = re.compile(
11    r"(?P<name>[\w]+)(?P<brackets>\[\])?(?P<delim>\.|::|->)?")
12
13
14def _split_param_ref(val):
15    return [name for name, _, _ in _PARAM_REF_NAME_RE.findall(val)]
16
17
18def _human_readable_deref(val, make_param_name=None):
19    """Turn the "name[].member[]" notation into plain English."""
20    parts = []
21    matches = _PARAM_REF_NAME_RE.findall(val)
22    for name, brackets, delim in reversed(matches):
23        if make_param_name:
24            name = make_param_name(name)
25        if delim:
26            parts.append('member of')
27        if brackets:
28            parts.append('each element of')
29        parts.append('the')
30        parts.append(name)
31    parts.append('parameter')
32    return ' '.join(parts)
33
34
35class LengthEntry:
36    """An entry in a (comma-separated) len attribute"""
37    NULL_TERMINATED_STRING = 'null-terminated'
38    MATH_STRING = 'latexmath:'
39
40    def __init__(self, val):
41        self.full_reference = val
42        self.other_param_name = None
43        self.null_terminated = False
44        self.number = None
45        self.math = None
46        self.param_ref_parts = None
47        if val == LengthEntry.NULL_TERMINATED_STRING:
48            self.null_terminated = True
49            return
50
51        if val.startswith(LengthEntry.MATH_STRING):
52            self.math = val.replace(LengthEntry.MATH_STRING, '')[1:-1]
53            return
54
55        if val.isdigit():
56            self.number = int(val)
57            return
58
59        # Must be another param name.
60        self.param_ref_parts = _split_param_ref(val)
61        self.other_param_name = self.param_ref_parts[0]
62
63    def __str__(self):
64        return self.full_reference
65
66    def get_human_readable(self, make_param_name=None):
67        assert(self.other_param_name)
68        return _human_readable_deref(self.full_reference, make_param_name=make_param_name)
69
70    def __repr__(self):
71        "Formats an object for repr(), debugger display, etc."
72        return 'spec_tools.attributes.LengthEntry("{}")'.format(self.full_reference)
73
74    @staticmethod
75    def parse_len_from_param(param):
76        """Get a list of LengthEntry, or None."""
77        len_str = param.get('len')
78        if len_str is None:
79            return None
80        return [LengthEntry(elt) for elt in len_str.split(',')]
81
82
83class ExternSyncEntry:
84    """An entry in a (comma-separated) externsync attribute"""
85
86    TRUE_STRING = 'true'
87    TRUE_WITH_CHILDREN_STRING = 'true_with_children'
88
89    def __init__(self, val):
90        self.full_reference = val
91        self.entirely_extern_sync = (val in (ExternSyncEntry.TRUE_STRING, ExternSyncEntry.TRUE_WITH_CHILDREN_STRING))
92        self.children_extern_sync = (val == ExternSyncEntry.TRUE_WITH_CHILDREN_STRING)
93        if self.entirely_extern_sync:
94            return
95
96        self.param_ref_parts = _split_param_ref(val)
97        self.member = self.param_ref_parts[0]
98
99    def get_human_readable(self, make_param_name=None):
100        assert(not self.entirely_extern_sync)
101        return _human_readable_deref(self.full_reference, make_param_name=make_param_name)
102
103    @staticmethod
104    def parse_externsync_from_param(param):
105        """Get a list of ExternSyncEntry."""
106        sync_str = param.get('externsync')
107        if sync_str is None:
108            return None
109        return [ExternSyncEntry(elt) for elt in sync_str.split(',')]
110
111    def __repr__(self):
112        "Formats an object for repr(), debugger display, etc."
113        return 'spec_tools.attributes.ExternSyncEntry("{}")'.format(self.full_reference)
114
115
116_TRUE_STRING = 'true'
117_FALSE_STRING = 'false'
118
119
120def _parse_optional_elt(val):
121    if val not in (_TRUE_STRING, _FALSE_STRING):
122        raise ValueError("Each element of the optional attribute must be 'true', or 'false'")
123    return val == _TRUE_STRING
124
125
126def parse_optional_from_param(param):
127    """Get a list of booleans from a param: always returns at least one element."""
128    optional_str = param.get('optional', _FALSE_STRING)
129    return [_parse_optional_elt(elt) for elt in optional_str.split(',')]
130
131
132def has_any_optional_in_param(param):
133    """Returns True if we have any true in an optional attribute."""
134    return any(parse_optional_from_param(param))
135