216 lines
6.1 KiB
Python
216 lines
6.1 KiB
Python
# SPDX-License-Identifier: GPL-2.0
|
|
#
|
|
# Radix Tree Parser
|
|
#
|
|
# Copyright (c) 2016 Linaro Ltd
|
|
# Copyright (c) 2023 Broadcom
|
|
#
|
|
# Authors:
|
|
# Kieran Bingham <kieran.bingham@linaro.org>
|
|
# Florian Fainelli <f.fainelli@gmail.com>
|
|
|
|
import gdb
|
|
|
|
from linux import utils
|
|
from linux import constants
|
|
|
|
radix_tree_root_type = utils.CachedType("struct xarray")
|
|
radix_tree_node_type = utils.CachedType("struct xa_node")
|
|
|
|
def is_internal_node(node):
|
|
long_type = utils.get_long_type()
|
|
return ((node.cast(long_type) & constants.LX_RADIX_TREE_ENTRY_MASK) == constants.LX_RADIX_TREE_INTERNAL_NODE)
|
|
|
|
def entry_to_node(node):
|
|
long_type = utils.get_long_type()
|
|
node_type = node.type
|
|
indirect_ptr = node.cast(long_type) & ~constants.LX_RADIX_TREE_INTERNAL_NODE
|
|
return indirect_ptr.cast(radix_tree_node_type.get_type().pointer())
|
|
|
|
def node_maxindex(node):
|
|
return (constants.LX_RADIX_TREE_MAP_SIZE << node['shift']) - 1
|
|
|
|
def resolve_root(root):
|
|
if root.type == radix_tree_root_type.get_type():
|
|
return root
|
|
if root.type == radix_tree_root_type.get_type().pointer():
|
|
return root.dereference()
|
|
raise gdb.GdbError("must be {} not {}"
|
|
.format(radix_tree_root_type.get_type(), root.type))
|
|
|
|
def lookup(root, index):
|
|
root = resolve_root(root)
|
|
node = root['xa_head']
|
|
if node == 0:
|
|
return None
|
|
|
|
if not (is_internal_node(node)):
|
|
if (index > 0):
|
|
return None
|
|
return node
|
|
|
|
node = entry_to_node(node)
|
|
maxindex = node_maxindex(node)
|
|
|
|
if (index > maxindex):
|
|
return None
|
|
|
|
shift = node['shift'] + constants.LX_RADIX_TREE_MAP_SHIFT
|
|
|
|
while True:
|
|
offset = (index >> node['shift']) & constants.LX_RADIX_TREE_MAP_MASK
|
|
slot = node['slots'][offset]
|
|
|
|
if slot == 0:
|
|
return None
|
|
|
|
node = slot.cast(node.type.pointer()).dereference()
|
|
if node == 0:
|
|
return None
|
|
|
|
shift -= constants.LX_RADIX_TREE_MAP_SHIFT
|
|
if (shift <= 0):
|
|
break
|
|
|
|
return node
|
|
|
|
def descend(parent, index):
|
|
offset = (index >> int(parent["shift"])) & constants.LX_RADIX_TREE_MAP_MASK
|
|
return offset, parent["slots"][offset]
|
|
|
|
def load_root(root):
|
|
node = root["xa_head"]
|
|
nodep = node
|
|
|
|
if is_internal_node(node):
|
|
node = entry_to_node(node)
|
|
maxindex = node_maxindex(node)
|
|
return int(node["shift"]) + constants.LX_RADIX_TREE_MAP_SHIFT, \
|
|
nodep, maxindex
|
|
|
|
return 0, nodep, 0
|
|
|
|
class RadixTreeIter:
|
|
def __init__(self, start):
|
|
self.index = 0
|
|
self.next_index = start
|
|
self.node = None
|
|
|
|
def xa_mk_internal(v):
|
|
return (v << 2) | 2
|
|
|
|
LX_XA_RETRY_ENTRY = xa_mk_internal(256)
|
|
LX_RADIX_TREE_RETRY = LX_XA_RETRY_ENTRY
|
|
|
|
def next_chunk(root, iter):
|
|
mask = (1 << (utils.get_ulong_type().sizeof * 8)) - 1
|
|
|
|
index = iter.next_index
|
|
if index == 0 and iter.index != 0:
|
|
return None
|
|
|
|
restart = True
|
|
while restart:
|
|
restart = False
|
|
|
|
_, child, maxindex = load_root(root)
|
|
if index > maxindex:
|
|
return None
|
|
if not child:
|
|
return None
|
|
|
|
if not is_internal_node(child):
|
|
iter.index = index
|
|
iter.next_index = (maxindex + 1) & mask
|
|
iter.node = None
|
|
return root["xa_head"].address
|
|
|
|
while True:
|
|
node = entry_to_node(child)
|
|
offset, child = descend(node, index)
|
|
|
|
if not child:
|
|
while True:
|
|
offset += 1
|
|
if offset >= constants.LX_RADIX_TREE_MAP_SIZE:
|
|
break
|
|
slot = node["slots"][offset]
|
|
if slot:
|
|
break
|
|
index &= ~node_maxindex(node)
|
|
index = (index + (offset << int(node["shift"]))) & mask
|
|
if index == 0:
|
|
return None
|
|
if offset == constants.LX_RADIX_TREE_MAP_SIZE:
|
|
restart = True
|
|
break
|
|
child = node["slots"][offset]
|
|
|
|
if not child:
|
|
restart = True
|
|
break
|
|
if child == LX_XA_RETRY_ENTRY:
|
|
break
|
|
if not node["shift"] or not is_internal_node(child):
|
|
break
|
|
|
|
iter.index = (index & ~node_maxindex(node)) | offset
|
|
iter.next_index = ((index | node_maxindex(node)) + 1) & mask
|
|
iter.node = node
|
|
|
|
return node["slots"][offset].address
|
|
|
|
def next_slot(slot, iter):
|
|
mask = (1 << (utils.get_ulong_type().sizeof * 8)) - 1
|
|
for _ in range(iter.next_index - iter.index - 1):
|
|
slot += 1
|
|
iter.index = (iter.index + 1) & mask
|
|
if slot.dereference():
|
|
return slot
|
|
return None
|
|
|
|
def for_each_slot(root, start=0):
|
|
iter = RadixTreeIter(start)
|
|
slot = None
|
|
while True:
|
|
if not slot:
|
|
slot = next_chunk(root, iter)
|
|
if not slot:
|
|
break
|
|
yield iter.index, slot
|
|
slot = next_slot(slot, iter)
|
|
|
|
class LxRadixTreeLookup(gdb.Function):
|
|
""" Lookup and return a node from a RadixTree.
|
|
|
|
$lx_radix_tree_lookup(root_node [, index]): Return the node at the given index.
|
|
If index is omitted, the root node is dereference and returned."""
|
|
|
|
def __init__(self):
|
|
super(LxRadixTreeLookup, self).__init__("lx_radix_tree_lookup")
|
|
|
|
def invoke(self, root, index=0):
|
|
result = lookup(root, index)
|
|
if result is None:
|
|
raise gdb.GdbError("No entry in tree at index {}".format(index))
|
|
|
|
return result
|
|
|
|
class LxRadixTree(gdb.Command):
|
|
"""Show all values stored in a RadixTree."""
|
|
|
|
def __init__(self):
|
|
super(LxRadixTree, self).__init__("lx-radix-tree", gdb.COMMAND_DATA,
|
|
gdb.COMPLETE_NONE)
|
|
|
|
def invoke(self, argument, from_tty):
|
|
args = gdb.string_to_argv(argument)
|
|
if len(args) != 1:
|
|
raise gdb.GdbError("Usage: lx-radix-tree ROOT")
|
|
root = gdb.parse_and_eval(args[0])
|
|
for index, slot in for_each_slot(root):
|
|
gdb.write("[{}] = {}\n".format(index, slot.dereference()))
|
|
|
|
LxRadixTree()
|
|
LxRadixTreeLookup()
|