Skip to content

Commit

Permalink
of: Make of_dma_get_range() work on bus nodes
Browse files Browse the repository at this point in the history
Since the "dma-ranges" property is only valid for a node representing a
bus, of_dma_get_range() currently assumes the node passed in is a leaf
representing a device, and starts the walk from its parent. In cases
like PCI host controllers on typical FDT systems, however, where the PCI
endpoints are probed dynamically the initial leaf node represents the
'bus' itself, and this logic means we fail to consider any "dma-ranges"
describing the host bridge itself. Rework the logic such that
of_dma_get_range() also works correctly starting from a bus node
containing "dma-ranges".

While this does mean "dma-ranges" could incorrectly be in a device leaf
node, there isn't really any way in this function to ensure that a leaf
node is or isn't a bus node.

Signed-off-by: Robin Murphy <[email protected]>
[robh: Allow for the bus child node to still be passed in]
Signed-off-by: Rob Herring <[email protected]>
Reviewed-by: Robin Murphy <[email protected]>
Reviewed-by: Nicolas Saenz Julienne <[email protected]>
Tested-by: Nicolas Saenz Julienne <[email protected]>
  • Loading branch information
rmurphy-arm authored and robherring committed Oct 9, 2019
1 parent 645c138 commit 951d488
Showing 1 changed file with 18 additions and 26 deletions.
44 changes: 18 additions & 26 deletions drivers/of/address.c
Original file line number Diff line number Diff line change
Expand Up @@ -930,55 +930,47 @@ int of_dma_get_range(struct device_node *np, u64 *dma_addr, u64 *paddr, u64 *siz
const __be32 *ranges = NULL;
int len, naddr, nsize, pna;
int ret = 0;
bool found_dma_ranges = false;
u64 dmaaddr;

if (!node)
return -EINVAL;

while (1) {
struct device_node *parent;

naddr = of_n_addr_cells(node);
nsize = of_n_size_cells(node);

parent = __of_get_dma_parent(node);
of_node_put(node);

node = parent;
if (!node)
break;

while (node) {
ranges = of_get_property(node, "dma-ranges", &len);

/* Ignore empty ranges, they imply no translation required */
if (ranges && len > 0)
break;

/*
* At least empty ranges has to be defined for parent node if
* DMA is supported
*/
if (!ranges)
break;
/* Once we find 'dma-ranges', then a missing one is an error */
if (found_dma_ranges && !ranges) {
ret = -ENODEV;
goto out;
}
found_dma_ranges = true;

node = of_get_next_dma_parent(node);
}

if (!ranges) {
if (!node || !ranges) {
pr_debug("no dma-ranges found for node(%pOF)\n", np);
ret = -ENODEV;
goto out;
}

len /= sizeof(u32);

naddr = of_bus_n_addr_cells(node);
nsize = of_bus_n_size_cells(node);
pna = of_n_addr_cells(node);
if ((len / sizeof(__be32)) % (pna + naddr + nsize)) {
ret = -EINVAL;
goto out;
}

/* dma-ranges format:
* DMA addr : naddr cells
* CPU addr : pna cells
* size : nsize cells
*/
dmaaddr = of_read_number(ranges, naddr);
*paddr = of_translate_dma_address(np, ranges);
*paddr = of_translate_dma_address(node, ranges + naddr);
if (*paddr == OF_BAD_ADDR) {
pr_err("translation of DMA address(%llx) to CPU address failed node(%pOF)\n",
dmaaddr, np);
Expand Down

0 comments on commit 951d488

Please sign in to comment.