int *nr_inport, int *nr_outport)
 {
        struct device_node *ep = NULL;
+       struct of_endpoint endpoint;
        int in = 0, out = 0;
 
        do {
                if (!ep)
                        break;
 
-               if (of_coresight_legacy_ep_is_input(ep))
-                       in++;
-               else
-                       out++;
+               if (of_graph_parse_endpoint(ep, &endpoint))
+                       continue;
+
+               if (of_coresight_legacy_ep_is_input(ep)) {
+                       in = (endpoint.port + 1 > in) ?
+                               endpoint.port + 1 : in;
+               } else {
+                       out = (endpoint.port + 1) > out ?
+                               endpoint.port + 1 : out;
+               }
 
        } while (ep);
 
 {
        int i = 0;
        struct device_node *ep = NULL;
+       struct of_endpoint endpoint;
+
+       while ((ep = of_graph_get_next_endpoint(port_parent, ep))) {
+               /* Defer error handling to parsing */
+               if (of_graph_parse_endpoint(ep, &endpoint))
+                       continue;
+               if (endpoint.port + 1 > i)
+                       i = endpoint.port + 1;
+       }
 
-       while ((ep = of_graph_get_next_endpoint(port_parent, ep)))
-               i++;
        return i;
 }
 
  * Parses the local port, remote device name and the remote port.
  *
  * Returns :
- *      1      - If the parsing is successful and a connection record
- *               was created for an output connection.
  *      0      - If the parsing completed without any fatal errors.
  *     -Errno  - Fatal error, abort the scanning.
  */
 static int of_coresight_parse_endpoint(struct device *dev,
                                       struct device_node *ep,
-                                      struct coresight_connection *conn)
+                                      struct coresight_platform_data *pdata)
 {
        int ret = 0;
        struct of_endpoint endpoint, rendpoint;
        struct device_node *rep = NULL;
        struct device *rdev = NULL;
        struct fwnode_handle *rdev_fwnode;
+       struct coresight_connection *conn;
 
        do {
                /* Parse the local port details */
                        break;
                }
 
+               conn = &pdata->conns[endpoint.port];
+               if (conn->child_fwnode) {
+                       dev_warn(dev, "Duplicate output port %d\n",
+                                endpoint.port);
+                       ret = -EINVAL;
+                       break;
+               }
                conn->outport = endpoint.port;
                /*
                 * Hold the refcount to the target device. This could be
                conn->child_fwnode = fwnode_handle_get(rdev_fwnode);
                conn->child_port = rendpoint.port;
                /* Connection record updated */
-               ret = 1;
        } while (0);
 
        of_node_put(rparent);
                                          struct coresight_platform_data *pdata)
 {
        int ret = 0;
-       struct coresight_connection *conn;
        struct device_node *ep = NULL;
        const struct device_node *parent = NULL;
        bool legacy_binding = false;
                dev_warn_once(dev, "Uses obsolete Coresight DT bindings\n");
        }
 
-       conn = pdata->conns;
-
        /* Iterate through each output port to discover topology */
        while ((ep = of_graph_get_next_endpoint(parent, ep))) {
                /*
                if (legacy_binding && of_coresight_legacy_ep_is_input(ep))
                        continue;
 
-               ret = of_coresight_parse_endpoint(dev, ep, conn);
-               switch (ret) {
-               case 1:
-                       conn++;         /* Fall through */
-               case 0:
-                       break;
-               default:
+               ret = of_coresight_parse_endpoint(dev, ep, pdata);
+               if (ret)
                        return ret;
-               }
        }
 
        return 0;
                 *    coresight_remove_match().
                 */
                conn->child_fwnode = fwnode_handle_get(&r_adev->fwnode);
+       } else if (dir == ACPI_CORESIGHT_LINK_SLAVE) {
+               /*
+                * We are only interested in the port number
+                * for the input ports at this component.
+                * Store the port number in child_port.
+                */
+               conn->child_port = fields[0].integer.value;
+       } else {
+               /* Invalid direction */
+               return -EINVAL;
        }
 
        return dir;
                        return dir;
 
                if (dir == ACPI_CORESIGHT_LINK_MASTER) {
-                       pdata->nr_outport++;
+                       if (ptr->outport > pdata->nr_outport)
+                               pdata->nr_outport = ptr->outport;
                        ptr++;
                } else {
-                       pdata->nr_inport++;
+                       WARN_ON(pdata->nr_inport == ptr->child_port);
+                       /*
+                        * We do not track input port connections for a device.
+                        * However we need the highest port number described,
+                        * which can be recorded now and reuse this connection
+                        * record for an output connection. Hence, do not move
+                        * the ptr for input connections
+                        */
+                       if (ptr->child_port > pdata->nr_inport)
+                               pdata->nr_inport = ptr->child_port;
                }
        }
 
                return rc;
 
        /* Copy the connection information to the final location */
-       for (i = 0; i < pdata->nr_outport; i++)
-               pdata->conns[i] = conns[i];
+       for (i = 0; conns + i < ptr; i++) {
+               int port = conns[i].outport;
+
+               /* Duplicate output port */
+               WARN_ON(pdata->conns[port].child_fwnode);
+               pdata->conns[port] = conns[i];
+       }
 
        devm_kfree(&adev->dev, conns);
        return 0;