Add support for the PCS to specify which interfaces it supports, which
can be used by MAC drivers to build the main supported_interfaces
bitmap. Phylink also validates that the PCS returned by the MAC driver
supports the interface that the MAC was asked for.
An empty supported_interfaces bitmap from the PCS indicates that it
does not provide this information, and we handle that appropriately.
Reviewed-by: Maxime Chevallier <maxime.chevallier@bootlin.com>
Signed-off-by: Russell King (Oracle) <rmk+kernel@armlinux.org.uk>
Link: https://patch.msgid.link/E1tTffL-007RoD-1Y@rmk-PC.armlinux.org.uk
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
                        return -EINVAL;
                }
 
+               /* Ensure that this PCS supports the interface which the MAC
+                * returned it for. It is an error for the MAC to return a PCS
+                * that does not support the interface mode.
+                */
+               if (!phy_interface_empty(pcs->supported_interfaces) &&
+                   !test_bit(state->interface, pcs->supported_interfaces)) {
+                       phylink_err(pl, "MAC returned PCS which does not support %s\n",
+                                   phy_modes(state->interface));
+                       return -EINVAL;
+               }
+
                /* Validate the link parameters with the PCS */
                if (pcs->ops->pcs_validate) {
                        ret = pcs->ops->pcs_validate(pcs, supported, state);
 
 
 /**
  * struct phylink_pcs - PHYLINK PCS instance
+ * @supported_interfaces: describing which PHY_INTERFACE_MODE_xxx
+ *                        are supported by this PCS.
  * @ops: a pointer to the &struct phylink_pcs_ops structure
  * @phylink: pointer to &struct phylink_config
  * @neg_mode: provide PCS neg mode via "mode" argument
  * the PCS driver.
  */
 struct phylink_pcs {
+       DECLARE_PHY_INTERFACE_MASK(supported_interfaces);
        const struct phylink_pcs_ops *ops;
        struct phylink *phylink;
        bool neg_mode;