blob: cd2ac0f9848c039f303640828cefa50335907aa8 [file] [log] [blame]
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright (C) 2022 F4PGA Authors
#
# 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.
#
# SPDX-License-Identifier: Apache-2.0
"""
Block hierarchy path utilities.
"""
import re
# =============================================================================
# A regular expression for parsing path nodes
PATH_NODE_RE = re.compile(r"^(?P<name>[^\s\[\]\.]+)(\[(?P<index>[0-9]+)\])?" r"(\[(?P<mode>[^\s\[\]\.]+)\])?$")
# =============================================================================
class PathNode:
"""
A class for representing a single path node.
A string representation of a path node can have the following formats:
- "<name>[<index>][<mode>]"
- "<name>[<index>]"
- "<name>[<mode>]"
- "<name>"
Differentiation between index and mode is based on the assumption that an
index may only contain numbers while a mode any other character as well.
FIXME: This may lead to ambiguity if a mode is named using digits only.
"""
def __init__(self, name, index=None, mode=None):
"""
Generic constructor
"""
assert isinstance(name, str) or name is None, name
assert isinstance(index, int) or index is None, index
assert isinstance(mode, str) or mode is None, mode
self.name = name
self.index = index
self.mode = mode
@staticmethod
def from_string(string):
"""
Converts a string representation to a PathNode object
"""
# Match the regex
match = PATH_NODE_RE.fullmatch(string)
assert match is not None, string
name = match.group("name")
index = match.group("index")
mode = match.group("mode")
if index is not None:
index = int(index)
return PathNode(name, index, mode)
def to_string(self):
"""
Converts the node into a string
"""
string = self.name
if self.index is not None:
string += "[{}]".format(self.index)
if self.mode is not None:
string += "[{}]".format(self.mode)
return string
def __str__(self):
return self.to_string()
def __repr__(self):
return self.to_string()