net: ptp: introduce .supported_extts_flags to ptp_clock_info
authorJacob Keller <jacob.e.keller@intel.com>
Mon, 14 Apr 2025 21:26:30 +0000 (14:26 -0700)
committerJakub Kicinski <kuba@kernel.org>
Wed, 16 Apr 2025 03:20:57 +0000 (20:20 -0700)
The PTP_EXTTS_REQUEST(2) ioctl has a flags field which specifies how the
external timestamp request should behave. This includes which edge of the
signal to timestamp, as well as a specialized "offset" mode. It is expected
that more flags will be added in the future.

Driver authors routinely do not check the flags, often accepting requests
with flags which they do not support. Even drivers which do check flags may
not be future-proofed to reject flags not yet defined. Thus, any future
flag additions often require manually updating drivers to reject these
flags.

This approach of hoping we catch flag checks during review, or playing
whack-a-mole after the fact is the wrong approach.

Introduce the "supported_extts_flags" field to the ptp_clock_info
structure. This field defines the set of flags the device actually
supports.

Update the core character device logic to check this field and reject
unsupported requests. Getting this right is somewhat tricky. First, to
avoid unnecessary repetition and make basic functionality work when
.supported_extts_flags is 0, the core always accepts the PTP_ENABLE_FEATURE
flag. This flag is used to set the 'on' parameter to the .enable function
and is thus always 'supported' by all drivers.

For backwards compatibility, the PTP_RISING_EDGE and PTP_FALLING_EDGE flags
are merely "hints" when using the old PTP_EXTTS_REQUEST ioctl, and are not
expected to be enforced. If the user issues PTP_EXTTS_REQUEST2, the
PTP_STRICT_FLAGS flag is added which is supposed to inform the driver to
strictly validate the flags and reject unsupported requests. To handle
this, first check if the driver reports PTP_STRICT_FLAGS support. If it
does not, then always allow the PTP_RISING_EDGE and PTP_FALLING_EDGE flags.
This keeps backwards compatibility with the original PTP_EXTTS_REQUEST
ioctl where these flags are not guaranteed to be honored.

This way, drivers which do not set the supported_extts_flags will continue
to accept requests for the original PTP_EXTTS_REQUEST ioctl. The core will
automatically reject requests with new flags, and correctly reject requests
with PTP_STRICT_FLAGS, where the driver is supposed to strictly validate
the flags.

Update the various drivers, refactoring their validation logic into the
.supported_extts_flags field. For consistency and readability,
PTP_ENABLE_FEATURE is not set in the supported flags list, and
PTP_EXTTS_EDGES is expanded to PTP_RISING_EDGE | PTP_FALLING_EDGE in all
cases.

Note the following driver files set n_ext_ts to a non-zero value but did
not check flags at all:

 • drivers/net/ethernet/freescale/dpaa2/dpaa2-ptp.c
 • drivers/net/ethernet/freescale/enetc/enetc_ptp.c
 • drivers/net/ethernet/intel/i40e/i40e_ptp.c
 • drivers/net/ethernet/marvell/octeontx2/nic/otx2_ptp.c
 • drivers/net/ethernet/renesas/ravb_ptp.c
 • drivers/net/ethernet/renesas/rtsn.c
 • drivers/net/ethernet/renesas/rtsn.h
 • drivers/net/ethernet/ti/am65-cpts.c
 • drivers/net/ethernet/ti/cpts.h
 • drivers/net/ethernet/ti/icssg/icss_iep.c
 • drivers/net/ethernet/xscale/ptp_ixp46x.c
 • drivers/net/phy/bcm-phy-ptp.c
 • drivers/ptp/ptp_ocp.c
 • drivers/ptp/ptp_pch.c
 • drivers/ptp/ptp_qoriq.c

These drivers behavior does change slightly: they will now reject the
PTP_EXTTS_REQUEST2 ioctl, because they do not strictly validate their
flags. This also makes them no longer incorrectly accept PTP_EXT_OFFSET.

Also note that the renesas ravb driver does not support PTP_STRICT_FLAGS.
We could leave the .supported_extts_flags as 0, but I added the
PTP_RISING_EDGE | PTP_FALLING_EDGE since the driver previously manually
validated these flags. This is equivalent to 0 because the core will allow
these flags regardless unless PTP_STRICT_FLAGS is also set.

Signed-off-by: Jacob Keller <jacob.e.keller@intel.com>
Reviewed-by: Kory Maincent <kory.maincent@bootlin.com>
Link: https://patch.msgid.link/20250414-jk-supported-perout-flags-v2-1-f6b17d15475c@intel.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
17 files changed:
drivers/net/dsa/mv88e6xxx/ptp.c
drivers/net/dsa/sja1105/sja1105_ptp.c
drivers/net/ethernet/intel/ice/ice_ptp.c
drivers/net/ethernet/intel/igb/igb_ptp.c
drivers/net/ethernet/intel/igc/igc_ptp.c
drivers/net/ethernet/mellanox/mlx5/core/lib/clock.c
drivers/net/ethernet/microchip/lan743x_ptp.c
drivers/net/ethernet/microchip/lan966x/lan966x_ptp.c
drivers/net/ethernet/renesas/ravb_ptp.c
drivers/net/phy/dp83640.c
drivers/net/phy/micrel.c
drivers/net/phy/nxp-c45-tja11xx.c
drivers/ptp/ptp_chardev.c
drivers/ptp/ptp_clockmatrix.c
drivers/ptp/ptp_fc3.c
drivers/ptp/ptp_idt82p33.c
include/linux/ptp_clock_kernel.h

index aed4a4b..1d3b2c9 100644 (file)
@@ -332,13 +332,6 @@ static int mv88e6352_ptp_enable_extts(struct mv88e6xxx_chip *chip,
        int pin;
        int err;
 
-       /* Reject requests with unsupported flags */
-       if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                               PTP_RISING_EDGE |
-                               PTP_FALLING_EDGE |
-                               PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
        /* Reject requests to enable time stamping on both edges. */
        if ((rq->extts.flags & PTP_STRICT_FLAGS) &&
            (rq->extts.flags & PTP_ENABLE_FEATURE) &&
@@ -566,6 +559,10 @@ int mv88e6xxx_ptp_setup(struct mv88e6xxx_chip *chip)
        chip->ptp_clock_info.verify     = ptp_ops->ptp_verify;
        chip->ptp_clock_info.do_aux_work = mv88e6xxx_hwtstamp_work;
 
+       chip->ptp_clock_info.supported_extts_flags = PTP_RISING_EDGE |
+                                                    PTP_FALLING_EDGE |
+                                                    PTP_STRICT_FLAGS;
+
        if (ptp_ops->set_ptp_cpu_port) {
                struct dsa_port *dp;
                int upstream = 0;
index 198e787..3b979d8 100644 (file)
@@ -820,13 +820,6 @@ static int sja1105_extts_enable(struct sja1105_private *priv,
        if (extts->index != 0)
                return -EOPNOTSUPP;
 
-       /* Reject requests with unsupported flags */
-       if (extts->flags & ~(PTP_ENABLE_FEATURE |
-                            PTP_RISING_EDGE |
-                            PTP_FALLING_EDGE |
-                            PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
        /* We can only enable time stamping on both edges, sadly. */
        if ((extts->flags & PTP_STRICT_FLAGS) &&
            (extts->flags & PTP_ENABLE_FEATURE) &&
@@ -912,6 +905,9 @@ int sja1105_ptp_clock_register(struct dsa_switch *ds)
                .n_pins         = 1,
                .n_ext_ts       = 1,
                .n_per_out      = 1,
+               .supported_extts_flags = PTP_RISING_EDGE |
+                                        PTP_FALLING_EDGE |
+                                        PTP_STRICT_FLAGS,
        };
 
        /* Only used on SJA1105 */
index 0ea5e52..660b321 100644 (file)
@@ -1627,14 +1627,6 @@ static int ice_ptp_cfg_extts(struct ice_pf *pf, struct ptp_extts_request *rq,
        int pin_desc_idx;
        u8 tmr_idx;
 
-       /* Reject requests with unsupported flags */
-
-       if (rq->flags & ~(PTP_ENABLE_FEATURE |
-                         PTP_RISING_EDGE |
-                         PTP_FALLING_EDGE |
-                         PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
        tmr_idx = hw->func_caps.ts_func_info.tmr_index_owned;
        chan = rq->index;
 
@@ -2740,6 +2732,10 @@ static void ice_ptp_set_caps(struct ice_pf *pf)
        info->enable = ice_ptp_gpio_enable;
        info->verify = ice_verify_pin;
 
+       info->supported_extts_flags = PTP_RISING_EDGE |
+                                     PTP_FALLING_EDGE |
+                                     PTP_STRICT_FLAGS;
+
        switch (pf->hw.mac_type) {
        case ICE_MAC_E810:
                ice_ptp_set_funcs_e810(pf);
index f323e1c..793c960 100644 (file)
@@ -502,13 +502,6 @@ static int igb_ptp_feature_enable_82580(struct ptp_clock_info *ptp,
 
        switch (rq->type) {
        case PTP_CLK_REQ_EXTTS:
-               /* Reject requests with unsupported flags */
-               if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                                       PTP_RISING_EDGE |
-                                       PTP_FALLING_EDGE |
-                                       PTP_STRICT_FLAGS))
-                       return -EOPNOTSUPP;
-
                /* Both the rising and falling edge are timestamped */
                if (rq->extts.flags & PTP_STRICT_FLAGS &&
                    (rq->extts.flags & PTP_ENABLE_FEATURE) &&
@@ -658,13 +651,6 @@ static int igb_ptp_feature_enable_i210(struct ptp_clock_info *ptp,
 
        switch (rq->type) {
        case PTP_CLK_REQ_EXTTS:
-               /* Reject requests with unsupported flags */
-               if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                                       PTP_RISING_EDGE |
-                                       PTP_FALLING_EDGE |
-                                       PTP_STRICT_FLAGS))
-                       return -EOPNOTSUPP;
-
                /* Reject requests failing to enable both edges. */
                if ((rq->extts.flags & PTP_STRICT_FLAGS) &&
                    (rq->extts.flags & PTP_ENABLE_FEATURE) &&
@@ -1356,6 +1342,9 @@ void igb_ptp_init(struct igb_adapter *adapter)
                adapter->ptp_caps.n_per_out = IGB_N_PEROUT;
                adapter->ptp_caps.n_pins = IGB_N_SDP;
                adapter->ptp_caps.pps = 0;
+               adapter->ptp_caps.supported_extts_flags = PTP_RISING_EDGE |
+                                                         PTP_FALLING_EDGE |
+                                                         PTP_STRICT_FLAGS;
                adapter->ptp_caps.pin_config = adapter->sdp_config;
                adapter->ptp_caps.adjfine = igb_ptp_adjfine_82580;
                adapter->ptp_caps.adjtime = igb_ptp_adjtime_82576;
@@ -1378,6 +1367,9 @@ void igb_ptp_init(struct igb_adapter *adapter)
                adapter->ptp_caps.n_ext_ts = IGB_N_EXTTS;
                adapter->ptp_caps.n_per_out = IGB_N_PEROUT;
                adapter->ptp_caps.n_pins = IGB_N_SDP;
+               adapter->ptp_caps.supported_extts_flags = PTP_RISING_EDGE |
+                                                         PTP_FALLING_EDGE |
+                                                         PTP_STRICT_FLAGS;
                adapter->ptp_caps.pps = 1;
                adapter->ptp_caps.pin_config = adapter->sdp_config;
                adapter->ptp_caps.adjfine = igb_ptp_adjfine_82580;
index 946edba..0c6c0cc 100644 (file)
@@ -257,13 +257,6 @@ static int igc_ptp_feature_enable_i225(struct ptp_clock_info *ptp,
 
        switch (rq->type) {
        case PTP_CLK_REQ_EXTTS:
-               /* Reject requests with unsupported flags */
-               if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                                       PTP_RISING_EDGE |
-                                       PTP_FALLING_EDGE |
-                                       PTP_STRICT_FLAGS))
-                       return -EOPNOTSUPP;
-
                /* Reject requests failing to enable both edges. */
                if ((rq->extts.flags & PTP_STRICT_FLAGS) &&
                    (rq->extts.flags & PTP_ENABLE_FEATURE) &&
@@ -1146,6 +1139,9 @@ void igc_ptp_init(struct igc_adapter *adapter)
                adapter->ptp_caps.pin_config = adapter->sdp_config;
                adapter->ptp_caps.n_ext_ts = IGC_N_EXTTS;
                adapter->ptp_caps.n_per_out = IGC_N_PEROUT;
+               adapter->ptp_caps.supported_extts_flags = PTP_RISING_EDGE |
+                                                         PTP_FALLING_EDGE |
+                                                         PTP_STRICT_FLAGS;
                adapter->ptp_caps.n_pins = IGC_N_SDP;
                adapter->ptp_caps.verify = igc_ptp_verify_pin;
 
index 65a94e4..3eee844 100644 (file)
@@ -643,13 +643,6 @@ static int mlx5_extts_configure(struct ptp_clock_info *ptp,
        int pin = -1;
        int err = 0;
 
-       /* Reject requests with unsupported flags */
-       if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                               PTP_RISING_EDGE |
-                               PTP_FALLING_EDGE |
-                               PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
        /* Reject requests to enable time stamping on both edges. */
        if ((rq->extts.flags & PTP_STRICT_FLAGS) &&
            (rq->extts.flags & PTP_ENABLE_FEATURE) &&
@@ -1034,6 +1027,10 @@ static void mlx5_init_pin_config(struct mlx5_core_dev *mdev)
        clock->ptp_info.verify = mlx5_ptp_verify;
        clock->ptp_info.pps = 1;
 
+       clock->ptp_info.supported_extts_flags = PTP_RISING_EDGE |
+                                               PTP_FALLING_EDGE |
+                                               PTP_STRICT_FLAGS;
+
        for (i = 0; i < clock->ptp_info.n_pins; i++) {
                snprintf(clock->ptp_info.pin_config[i].name,
                         sizeof(clock->ptp_info.pin_config[i].name),
index 0be44dc..b171c89 100644 (file)
@@ -942,12 +942,6 @@ static int lan743x_ptp_io_extts(struct lan743x_adapter *adapter, int on,
 
        extts = &ptp->extts[index];
 
-       if (extts_request->flags & ~(PTP_ENABLE_FEATURE |
-                                    PTP_RISING_EDGE |
-                                    PTP_FALLING_EDGE |
-                                    PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
        if (on) {
                extts_pin = ptp_find_pin(ptp->ptp_clock, PTP_PF_EXTTS, index);
                if (extts_pin < 0)
@@ -1543,6 +1537,9 @@ int lan743x_ptp_open(struct lan743x_adapter *adapter)
        ptp->ptp_clock_info.n_per_out = LAN743X_PTP_N_EVENT_CHAN;
        ptp->ptp_clock_info.n_pins = n_pins;
        ptp->ptp_clock_info.pps = LAN743X_PTP_N_PPS;
+       ptp->ptp_clock_info.supported_extts_flags = PTP_RISING_EDGE |
+                                                   PTP_FALLING_EDGE |
+                                                   PTP_STRICT_FLAGS;
        ptp->ptp_clock_info.pin_config = ptp->pin_config;
        ptp->ptp_clock_info.adjfine = lan743x_ptpci_adjfine;
        ptp->ptp_clock_info.adjtime = lan743x_ptpci_adjtime;
index 63905bb..8cf41b0 100644 (file)
@@ -917,12 +917,6 @@ static int lan966x_ptp_extts(struct ptp_clock_info *ptp,
        if (lan966x->ptp_ext_irq <= 0)
                return -EOPNOTSUPP;
 
-       /* Reject requests with unsupported flags */
-       if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                               PTP_RISING_EDGE |
-                               PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
        pin = ptp_find_pin(phc->clock, PTP_PF_EXTTS, rq->extts.index);
        if (pin == -1 || pin >= LAN966X_PHC_PINS_NUM)
                return -EINVAL;
@@ -978,6 +972,8 @@ static struct ptp_clock_info lan966x_ptp_clock_info = {
        .n_per_out      = LAN966X_PHC_PINS_NUM,
        .n_ext_ts       = LAN966X_PHC_PINS_NUM,
        .n_pins         = LAN966X_PHC_PINS_NUM,
+       .supported_extts_flags = PTP_RISING_EDGE |
+                                PTP_STRICT_FLAGS,
 };
 
 static int lan966x_ptp_phc_init(struct lan966x *lan966x,
index b436590..ab1ffdc 100644 (file)
@@ -176,12 +176,6 @@ static int ravb_ptp_extts(struct ptp_clock_info *ptp,
        struct net_device *ndev = priv->ndev;
        unsigned long flags;
 
-       /* Reject requests with unsupported flags */
-       if (req->flags & ~(PTP_ENABLE_FEATURE |
-                          PTP_RISING_EDGE |
-                          PTP_FALLING_EDGE))
-               return -EOPNOTSUPP;
-
        if (req->index)
                return -EINVAL;
 
@@ -287,6 +281,7 @@ static const struct ptp_clock_info ravb_ptp_info = {
        .max_adj        = 50000000,
        .n_ext_ts       = N_EXT_TS,
        .n_per_out      = N_PER_OUT,
+       .supported_extts_flags = PTP_RISING_EDGE | PTP_FALLING_EDGE,
        .adjfine        = ravb_ptp_adjfine,
        .adjtime        = ravb_ptp_adjtime,
        .gettime64      = ravb_ptp_gettime64,
index 85e2314..c89c255 100644 (file)
@@ -478,13 +478,6 @@ static int ptp_dp83640_enable(struct ptp_clock_info *ptp,
 
        switch (rq->type) {
        case PTP_CLK_REQ_EXTTS:
-               /* Reject requests with unsupported flags */
-               if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                                       PTP_RISING_EDGE |
-                                       PTP_FALLING_EDGE |
-                                       PTP_STRICT_FLAGS))
-                       return -EOPNOTSUPP;
-
                /* Reject requests to enable time stamping on both edges. */
                if ((rq->extts.flags & PTP_STRICT_FLAGS) &&
                    (rq->extts.flags & PTP_ENABLE_FEATURE) &&
@@ -1002,6 +995,9 @@ static void dp83640_clock_init(struct dp83640_clock *clock, struct mii_bus *bus)
        clock->caps.n_per_out   = N_PER_OUT;
        clock->caps.n_pins      = DP83640_N_PINS;
        clock->caps.pps         = 0;
+       clock->caps.supported_extts_flags = PTP_RISING_EDGE |
+                                           PTP_FALLING_EDGE |
+                                           PTP_STRICT_FLAGS;
        clock->caps.adjfine     = ptp_dp83640_adjfine;
        clock->caps.adjtime     = ptp_dp83640_adjtime;
        clock->caps.gettime64   = ptp_dp83640_gettime;
index 24882d3..61123ec 100644 (file)
@@ -3406,11 +3406,6 @@ static int lan8814_ptp_extts(struct ptp_clock_info *ptpci,
        struct phy_device *phydev = shared->phydev;
        int pin;
 
-       if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                               PTP_EXTTS_EDGES |
-                               PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
        pin = ptp_find_pin(shared->ptp_clock, PTP_PF_EXTTS,
                           rq->extts.index);
        if (pin == -1 || pin != LAN8814_PTP_EXTTS_NUM)
@@ -3917,6 +3912,9 @@ static int lan8814_ptp_probe_once(struct phy_device *phydev)
        shared->ptp_clock_info.n_ext_ts = LAN8814_PTP_EXTTS_NUM;
        shared->ptp_clock_info.n_pins = LAN8814_PTP_GPIO_NUM;
        shared->ptp_clock_info.pps = 0;
+       shared->ptp_clock_info.supported_extts_flags = PTP_RISING_EDGE |
+                                                      PTP_FALLING_EDGE |
+                                                      PTP_STRICT_FLAGS;
        shared->ptp_clock_info.pin_config = shared->pin_config;
        shared->ptp_clock_info.n_per_out = LAN8814_PTP_PEROUT_NUM;
        shared->ptp_clock_info.adjfine = lan8814_ptpci_adjfine;
index 250a018..8634b4c 100644 (file)
@@ -861,12 +861,6 @@ static int nxp_c45_extts_enable(struct nxp_c45_phy *priv,
        const struct nxp_c45_phy_data *data = nxp_c45_get_data(priv->phydev);
        int pin;
 
-       if (extts->flags & ~(PTP_ENABLE_FEATURE |
-                             PTP_RISING_EDGE |
-                             PTP_FALLING_EDGE |
-                             PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
        /* Sampling on both edges is not supported */
        if ((extts->flags & PTP_RISING_EDGE) &&
            (extts->flags & PTP_FALLING_EDGE) &&
@@ -962,6 +956,9 @@ static int nxp_c45_init_ptp_clock(struct nxp_c45_phy *priv)
                .n_pins         = ARRAY_SIZE(nxp_c45_ptp_pins),
                .n_ext_ts       = 1,
                .n_per_out      = 1,
+               .supported_extts_flags = PTP_RISING_EDGE |
+                                        PTP_FALLING_EDGE |
+                                        PTP_STRICT_FLAGS,
        };
 
        priv->ptp_clock = ptp_clock_register(&priv->caps,
index 4380e6d..c24228c 100644 (file)
@@ -162,6 +162,7 @@ long ptp_ioctl(struct posix_clock_context *pccontext, unsigned int cmd,
 {
        struct ptp_clock *ptp =
                container_of(pccontext->clk, struct ptp_clock, clock);
+       unsigned int i, pin_index, supported_extts_flags;
        struct ptp_sys_offset_extended *extoff = NULL;
        struct ptp_sys_offset_precise precise_offset;
        struct system_device_crosststamp xtstamp;
@@ -172,7 +173,6 @@ long ptp_ioctl(struct posix_clock_context *pccontext, unsigned int cmd,
        struct ptp_clock_request req;
        struct ptp_clock_caps caps;
        struct ptp_clock_time *pct;
-       unsigned int i, pin_index;
        struct ptp_pin_desc pd;
        struct timespec64 ts;
        int enable, err = 0;
@@ -240,6 +240,18 @@ long ptp_ioctl(struct posix_clock_context *pccontext, unsigned int cmd,
                        err = -EINVAL;
                        break;
                }
+               supported_extts_flags = ptp->info->supported_extts_flags;
+               /* The PTP_ENABLE_FEATURE flag is always supported. */
+               supported_extts_flags |= PTP_ENABLE_FEATURE;
+               /* If the driver does not support strictly checking flags, the
+                * PTP_RISING_EDGE and PTP_FALLING_EDGE flags are merely
+                * hints which are not enforced.
+                */
+               if (!(supported_extts_flags & PTP_STRICT_FLAGS))
+                       supported_extts_flags |= PTP_EXTTS_EDGES;
+               /* Reject unsupported flags */
+               if (req.extts.flags & ~supported_extts_flags)
+                       return -EOPNOTSUPP;
                req.type = PTP_CLK_REQ_EXTTS;
                enable = req.extts.flags & PTP_ENABLE_FEATURE ? 1 : 0;
                if (mutex_lock_interruptible(&ptp->pincfg_mux))
index fbb3fa8..b8d4df8 100644 (file)
@@ -283,18 +283,6 @@ static int idtcm_extts_enable(struct idtcm_channel *channel,
        idtcm = channel->idtcm;
        old_mask = idtcm->extts_mask;
 
-       /* Reject requests with unsupported flags */
-       if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                               PTP_RISING_EDGE |
-                               PTP_FALLING_EDGE |
-                               PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
-       /* Reject requests to enable time stamping on falling edge */
-       if ((rq->extts.flags & PTP_ENABLE_FEATURE) &&
-           (rq->extts.flags & PTP_FALLING_EDGE))
-               return -EOPNOTSUPP;
-
        if (index >= MAX_TOD)
                return -EINVAL;
 
@@ -2043,6 +2031,7 @@ static const struct ptp_clock_info idtcm_caps = {
        .n_per_out      = 12,
        .n_ext_ts       = MAX_TOD,
        .n_pins         = MAX_REF_CLK,
+       .supported_extts_flags = PTP_RISING_EDGE | PTP_STRICT_FLAGS,
        .adjphase       = &idtcm_adjphase,
        .getmaxphase    = &idtcm_getmaxphase,
        .adjfine        = &idtcm_adjfine,
@@ -2060,6 +2049,7 @@ static const struct ptp_clock_info idtcm_caps_deprecated = {
        .n_per_out      = 12,
        .n_ext_ts       = MAX_TOD,
        .n_pins         = MAX_REF_CLK,
+       .supported_extts_flags = PTP_RISING_EDGE | PTP_STRICT_FLAGS,
        .adjphase       = &idtcm_adjphase,
        .getmaxphase    = &idtcm_getmaxphase,
        .adjfine        = &idtcm_adjfine,
index cfced36..7000250 100644 (file)
@@ -592,6 +592,7 @@ static const struct ptp_clock_info idtfc3_caps = {
        .max_adj        = MAX_FFO_PPB,
        .n_per_out      = 1,
        .n_ext_ts       = 1,
+       .supported_extts_flags = PTP_STRICT_FLAGS | PTP_EXT_OFFSET,
        .adjphase       = &idtfc3_adjphase,
        .adjfine        = &idtfc3_adjfine,
        .adjtime        = &idtfc3_adjtime,
index b2fd94d..f01c50d 100644 (file)
@@ -246,18 +246,6 @@ static int idt82p33_extts_enable(struct idt82p33_channel *channel,
        idt82p33  = channel->idt82p33;
        old_mask = idt82p33->extts_mask;
 
-       /* Reject requests with unsupported flags */
-       if (rq->extts.flags & ~(PTP_ENABLE_FEATURE |
-                               PTP_RISING_EDGE |
-                               PTP_FALLING_EDGE |
-                               PTP_STRICT_FLAGS))
-               return -EOPNOTSUPP;
-
-       /* Reject requests to enable time stamping on falling edge */
-       if ((rq->extts.flags & PTP_ENABLE_FEATURE) &&
-           (rq->extts.flags & PTP_FALLING_EDGE))
-               return -EOPNOTSUPP;
-
        if (index >= MAX_PHC_PLL)
                return -EINVAL;
 
@@ -1187,6 +1175,9 @@ static void idt82p33_caps_init(u32 index, struct ptp_clock_info *caps,
 
        caps->pin_config = pin_cfg;
 
+       caps->supported_extts_flags = PTP_RISING_EDGE |
+                                     PTP_STRICT_FLAGS;
+
        for (i = 0; i < max_pins; ++i) {
                ppd = &pin_cfg[i];
 
index 0d68d09..25cba2e 100644 (file)
@@ -68,6 +68,17 @@ struct ptp_system_timestamp {
  * @n_per_out: The number of programmable periodic signals.
  * @n_pins:    The number of programmable pins.
  * @pps:       Indicates whether the clock supports a PPS callback.
+ *
+ * @supported_extts_flags:  The set of flags the driver supports for the
+ *                          PTP_EXTTS_REQUEST ioctl. The PTP core will use
+ *                          this list to reject unsupported requests.
+ *                          PTP_ENABLE_FEATURE is assumed and does not need to
+ *                          be included. If PTP_STRICT_FLAGS is *not* set,
+ *                          then both PTP_RISING_EDGE and PTP_FALLING_EDGE
+ *                          will be assumed. Note that PTP_STRICT_FLAGS must
+ *                          be set if the drivers wants to honor
+ *                          PTP_EXTTS_REQUEST2 and any future flags.
+ *
  * @pin_config: Array of length 'n_pins'. If the number of
  *              programmable pins is nonzero, then drivers must
  *              allocate and initialize this array.
@@ -174,6 +185,7 @@ struct ptp_clock_info {
        int n_per_out;
        int n_pins;
        int pps;
+       unsigned int supported_extts_flags;
        struct ptp_pin_desc *pin_config;
        int (*adjfine)(struct ptp_clock_info *ptp, long scaled_ppm);
        int (*adjphase)(struct ptp_clock_info *ptp, s32 phase);