On Thu, Jan 22, 2026 at 07:04:47PM -0500, Soumyadeep Hore wrote:
> Add a new file - idpf_ptp - to handle PTP virtchnl messages.
> Keep the registers addresses in the PTP struct.
> 
> Signed-off-by: Soumyadeep Hore <[email protected]>
> ---

Some review feedback inline below. See also review feedback from AI review
that was posted by Stephen.

>  drivers/net/intel/idpf/idpf_common_device.h   |   4 +
>  drivers/net/intel/idpf/idpf_common_virtchnl.c |  34 +-
>  drivers/net/intel/idpf/idpf_ptp.c             | 639 ++++++++++++++++++
>  drivers/net/intel/idpf/idpf_ptp.h             | 227 +++++++
>  drivers/net/intel/idpf/meson.build            |   1 +
>  5 files changed, 903 insertions(+), 2 deletions(-)
>  create mode 100644 drivers/net/intel/idpf/idpf_ptp.c
>  create mode 100644 drivers/net/intel/idpf/idpf_ptp.h
> 
> diff --git a/drivers/net/intel/idpf/idpf_common_device.h 
> b/drivers/net/intel/idpf/idpf_common_device.h
> index c32dcfbb12..3c84b96225 100644
> --- a/drivers/net/intel/idpf/idpf_common_device.h
> +++ b/drivers/net/intel/idpf/idpf_common_device.h
> @@ -90,6 +90,7 @@ struct idpf_adapter {
>  
>       /* For timestamp */
>       uint64_t time_hw;
> +     struct idpf_ptp *ptp;
>  
>       enum idpf_rx_func_type rx_func_type;
>  };
> @@ -161,6 +162,9 @@ struct idpf_vport {
>       /* Event from ipf */
>       bool link_up;
>       uint32_t link_speed;
> +
> +     /* For PTP */
> +     struct idpf_ptp_vport_tx_tstamp_caps *tx_tstamp_caps;
>  };
>  
>  /* Message type read in virtual channel from PF */
> diff --git a/drivers/net/intel/idpf/idpf_common_virtchnl.c 
> b/drivers/net/intel/idpf/idpf_common_virtchnl.c
> index e927d7415a..a28bc9bee6 100644
> --- a/drivers/net/intel/idpf/idpf_common_virtchnl.c
> +++ b/drivers/net/intel/idpf/idpf_common_virtchnl.c
> @@ -4,6 +4,7 @@
>  
>  #include "idpf_common_virtchnl.h"
>  #include "idpf_common_logs.h"
> +#include "idpf_ptp.h"
>  
>  #include <eal_export.h>
>  
> @@ -38,6 +39,28 @@ idpf_vc_clean(struct idpf_adapter *adapter)
>       return 0;
>  }
>  
> +/**
> + * idpf_ptp_is_mb_msg - Check if the message is PTP-related
> + * @op: virtchnl opcode
> + *
> + * Returns true if msg is PTP-related, false otherwise
> + */
> +static inline bool idpf_ptp_is_mb_msg(uint32_t op)
> +{

The name of this function seems backward to me. We are not returning if
it's a mailbox message, we are returning if it's PTP related, no? Therefore
would idpf_mb_msg_is_ptp, not be a better name?

> +     switch (op) {
> +     case VIRTCHNL2_OP_PTP_GET_VPORT_TX_TSTAMP:
> +     case VIRTCHNL2_OP_PTP_GET_DEV_CLK_TIME:
> +     case VIRTCHNL2_OP_PTP_GET_CROSS_TIME:
> +     case VIRTCHNL2_OP_PTP_SET_DEV_CLK_TIME:
> +     case VIRTCHNL2_OP_PTP_ADJ_DEV_CLK_FINE:
> +     case VIRTCHNL2_OP_PTP_ADJ_DEV_CLK_TIME:
> +     case VIRTCHNL2_OP_PTP_GET_VPORT_TX_TSTAMP_CAPS:
> +             return true;
> +     default:
> +             return false;
> +     }
> +}
> +
>  static int
>  idpf_send_vc_msg(struct idpf_adapter *adapter, uint32_t op,
>                uint16_t msg_size, uint8_t *msg)
> @@ -71,8 +94,15 @@ idpf_send_vc_msg(struct idpf_adapter *adapter, uint32_t op,
>  
>       memcpy(dma_mem->va, msg, msg_size);
>  
> -     ctlq_msg->opcode = idpf_mbq_opc_send_msg_to_pf;
> -     ctlq_msg->func_id = 0;
> +     if (idpf_ptp_is_mb_msg(op) && adapter->ptp->secondary_mbx.valid) {

If the first of these is true and the second not, is that an error
condition and do we need to handle it?

> +             ctlq_msg->opcode = idpf_mbq_opc_send_msg_to_peer_drv;
> +             ctlq_msg->func_id = adapter->ptp->secondary_mbx.peer_mbx_q_id;
> +             ctlq_msg->host_id = adapter->ptp->secondary_mbx.peer_id;
> +     } else {
> +             ctlq_msg->opcode = idpf_mbq_opc_send_msg_to_pf;
> +             ctlq_msg->func_id = 0;
> +     }
> +
>       ctlq_msg->data_len = msg_size;
>       ctlq_msg->cookie.mbx.chnl_opcode = op;
>       ctlq_msg->cookie.mbx.chnl_retval = VIRTCHNL_STATUS_SUCCESS;
> diff --git a/drivers/net/intel/idpf/idpf_ptp.c 
> b/drivers/net/intel/idpf/idpf_ptp.c
> new file mode 100644
> index 0000000000..b8f6afa518
> --- /dev/null
> +++ b/drivers/net/intel/idpf/idpf_ptp.c
> @@ -0,0 +1,639 @@
> +/* SPDX-License-Identifier: BSD-3-Clause
> + * Copyright(c) 2025 Intel Corporation
> + */
> +
> +#include "idpf_ptp.h"
> +#include <base/virtchnl2.h>
> +#include "idpf_common_virtchnl.h"
> +
> +/**
> + * idpf_ptp_get_access - Determine the access type of the PTP features
> + * @adapter: Driver specific private structure
> + * @direct: Capability that indicates the direct access
> + * @mailbox: Capability that indicates the mailbox access
> + *
> + * Return: the type of supported access for the PTP feature.
> + */
> +static enum idpf_ptp_access
> +idpf_ptp_get_access(const struct idpf_adapter *adapter, u32 direct, u32 
> mailbox)
> +{
> +     if (adapter->ptp->caps & direct)
> +             return IDPF_PTP_DIRECT;
> +     else if (adapter->ptp->caps & mailbox)
> +             return IDPF_PTP_MAILBOX;
> +     else
> +             return IDPF_PTP_NONE;
> +}
> +
> +/**
> + * idpf_ptp_get_features_access - Determine the access type of PTP features
> + * @adapter: Driver specific private structure
> + *
> + * Fulfill the adapter structure with type of the supported PTP features
> + * access.
> + */
> +static void idpf_ptp_get_features_access(const struct idpf_adapter *adapter)
> +{
> +     struct idpf_ptp *ptp = adapter->ptp;
> +     u32 direct, mailbox;
> +
> +     /* Get the device clock time */
> +     direct = VIRTCHNL2_CAP_PTP_GET_DEVICE_CLK_TIME;
> +     mailbox = VIRTCHNL2_CAP_PTP_GET_DEVICE_CLK_TIME_MB;
> +     ptp->get_dev_clk_time_access = (uint8_t)idpf_ptp_get_access(adapter,
> +                                                      direct,
> +                                                      mailbox);
> +
> +     /* Get the cross timestamp */
> +     direct = VIRTCHNL2_CAP_PTP_GET_CROSS_TIME;
> +     mailbox = VIRTCHNL2_CAP_PTP_GET_CROSS_TIME_MB;
> +     ptp->get_cross_tstamp_access = (uint8_t)idpf_ptp_get_access(adapter,
> +                                                      direct,
> +                                                      mailbox);
> +
> +     /* Set the device clock time */
> +     direct = VIRTCHNL2_CAP_PTP_SET_DEVICE_CLK_TIME;
> +     mailbox = VIRTCHNL2_CAP_PTP_SET_DEVICE_CLK_TIME_MB;
> +     ptp->set_dev_clk_time_access = (uint8_t)idpf_ptp_get_access(adapter,
> +                                                      direct,
> +                                                      mailbox);
> +
> +     /* Adjust the device clock time */
> +     direct = VIRTCHNL2_CAP_PTP_ADJ_DEVICE_CLK;
> +     mailbox = VIRTCHNL2_CAP_PTP_ADJ_DEVICE_CLK_MB;
> +     ptp->adj_dev_clk_time_access = (uint8_t)idpf_ptp_get_access(adapter,
> +                                                      direct,
> +                                                      mailbox);
> +
> +     /* Tx timestamping */
> +     direct = VIRTCHNL2_CAP_PTP_TX_TSTAMPS;
> +     mailbox = VIRTCHNL2_CAP_PTP_TX_TSTAMPS_MB;
> +     ptp->tx_tstamp_access = (uint8_t)idpf_ptp_get_access(adapter,
> +                                                      direct,
> +                                                      mailbox);
> +}

This function could be much shorter if you just removed the variables "direct"
and "mailbox". Rather than doing 

        direct = VIRTCHNL2_CAP_PTP_TX_TSTAMPS;
        mailbox = VIRTCHNL2_CAP_PTP_TX_TSTAMPS_MB;
        ptp->tx_tstamp_access = (uint8_t)idpf_ptp_get_access(adapter,
                                                         direct,
                                                         mailbox);

You can shorten the 5 lines down to 3 as:

        ptp->tx_tstamp_access = idpf_ptp_get_access(adapter,
                        VIRTCHNL2_CAP_PTP_TX_TSTAMPS,
                        VIRTCHNL2_CAP_PTP_TX_TSTAMPS_MB);

Furthermore, since all the defines used for mailbox are the same as those
used for direct, except for an _MB suffix, you could use a macro to shorten
thngs further if you want - maybe even fit each assignment on  one line.
For example (untested, and also shortening adapter to ad to fit one line):

#define IDPF_PTP_ACCESS(a, d) (uint8_t)idpf_ptp_get_access(a, d, d##_MB)

static void
idpf_ptp_get_features_access(const struct idpf_adapter *ad)
{
        struct idpf_ptp *ptp = ad->ptp;

        ptp->get_dev_clk_time_access = IDPF_PTP_ACCESS(ad, 
VIRTCHNL2_CAP_PTP_GET_DEVICE_CLK_TIME);
        ptp->get_cross_tstamp_access = IDPF_PTP_ACCESS(ad, 
VIRTCHNL2_CAP_PTP_GET_CROSS_TIME);
        ptp->set_dev_clk_time_access = IDPF_PTP_ACCESS(ad, 
VIRTCHNL2_CAP_PTP_SET_DEVICE_CLK_TIME);
        ptp->adj_dev_clk_time_access = IDPF_PTP_ACCESS(ad, 
VIRTCHNL2_CAP_PTP_ADJ_DEVICE_CLK);
        ptp->tx_stamp_access = IDPF_PTP_ACCESS(ad, 
VIRTCHNL2_CAP_PTP_TX_TSTAMPS);
}

Much shorter! :-)


> +/**
> + * idpf_ptp_get_caps - Send virtchnl get ptp capabilities message
> + * @adapter: Driver specific private structure
> + *
> + * Send virtchnl get PTP capabilities message.
> + *
> + * Return: 0 on success, -errno on failure.
> + */
> +int idpf_ptp_get_caps(struct idpf_adapter *adapter)
> +{
> +     struct virtchnl2_ptp_cross_time_reg_offsets cross_tstamp_offsets;
> +     struct virtchnl2_ptp_clk_adj_reg_offsets clk_adj_offsets;
> +     struct virtchnl2_ptp_get_caps send_ptp_caps_msg = { };

Using {} is a GNU extension to C11, so its best to use {0} as initializer.

> +     struct virtchnl2_ptp_clk_reg_offsets clock_offsets;
> +     struct virtchnl2_ptp_get_caps *recv_ptp_caps_msg;
> +     struct idpf_cmd_info args = { };
> +     struct idpf_ptp_secondary_mbx *scnd_mbx;
> +     struct idpf_ptp *ptp = adapter->ptp;
> +     struct idpf_hw *hw = &adapter->hw;
> +     enum idpf_ptp_access access_type;
> +     int err;
> +     u32 temp_offset;
> +
> +     send_ptp_caps_msg.caps = 
> CPU_TO_LE32(VIRTCHNL2_CAP_PTP_GET_DEVICE_CLK_TIME |
> +                                          
> VIRTCHNL2_CAP_PTP_GET_DEVICE_CLK_TIME_MB |
> +                                          VIRTCHNL2_CAP_PTP_GET_CROSS_TIME |
> +                                          
> VIRTCHNL2_CAP_PTP_GET_CROSS_TIME_MB |
> +                                          
> VIRTCHNL2_CAP_PTP_SET_DEVICE_CLK_TIME |
> +                                          
> VIRTCHNL2_CAP_PTP_SET_DEVICE_CLK_TIME_MB |
> +                                          VIRTCHNL2_CAP_PTP_ADJ_DEVICE_CLK |
> +                                          
> VIRTCHNL2_CAP_PTP_ADJ_DEVICE_CLK_MB |
> +                                          VIRTCHNL2_CAP_PTP_TX_TSTAMPS |
> +                                          VIRTCHNL2_CAP_PTP_TX_TSTAMPS_MB);
> +
> +     args.ops = VIRTCHNL2_OP_PTP_GET_CAPS;
> +     args.in_args = (uint8_t *)&send_ptp_caps_msg;
> +     args.in_args_size = sizeof(send_ptp_caps_msg);
> +
> +     args.out_buffer = adapter->mbx_resp;
> +     args.out_size = sizeof(*recv_ptp_caps_msg);
> +
> +     err = idpf_vc_cmd_execute(adapter, &args);
> +     if (err < 0)
> +             return err;
> +
> +     recv_ptp_caps_msg = (struct virtchnl2_ptp_get_caps *)args.out_buffer;
> +     ptp->caps = LE32_TO_CPU(recv_ptp_caps_msg->caps);
> +     ptp->base_incval = LE64_TO_CPU(recv_ptp_caps_msg->base_incval);
> +     ptp->max_adj = LE32_TO_CPU(recv_ptp_caps_msg->max_adj);
> +
> +     scnd_mbx = &ptp->secondary_mbx;
> +     scnd_mbx->peer_mbx_q_id = LE16_TO_CPU(recv_ptp_caps_msg->peer_mbx_q_id);
> +
> +     /* if the ptp_mb_q_id holds invalid value (0xffff), the secondary
> +      * mailbox is not supported.
> +      */
> +     scnd_mbx->valid = scnd_mbx->peer_mbx_q_id != 0xffff;
> +     if (scnd_mbx->valid)
> +             scnd_mbx->peer_id = recv_ptp_caps_msg->peer_id;
> +
> +     /* Determine the access type for the PTP features */
> +     idpf_ptp_get_features_access(adapter);
> +
> +     access_type = (enum idpf_ptp_access)ptp->get_dev_clk_time_access;
> +     if (access_type != IDPF_PTP_DIRECT)
> +             goto cross_tstamp;
> +
> +     clock_offsets = recv_ptp_caps_msg->clk_offsets;
> +
> +     temp_offset = LE32_TO_CPU(clock_offsets.dev_clk_ns_l);
> +     ptp->dev_clk_regs.dev_clk_ns_l = IDPF_PCI_REG_ADDR(hw,
> +                                                        temp_offset);

No need to wrap these lines. Use the full 100 chars.
(Also, when wrapping, double tab indent is sufficient, no need to put in 7
tabs + spaces to try and align braces in cases like this)

> +     temp_offset = LE32_TO_CPU(clock_offsets.dev_clk_ns_h);
> +     ptp->dev_clk_regs.dev_clk_ns_h = IDPF_PCI_REG_ADDR(hw,
> +                                                        temp_offset);
> +     temp_offset = LE32_TO_CPU(clock_offsets.phy_clk_ns_l);
> +     ptp->dev_clk_regs.phy_clk_ns_l = IDPF_PCI_REG_ADDR(hw,
> +                               temp_offset);
> +     temp_offset = LE32_TO_CPU(clock_offsets.phy_clk_ns_h);
> +     ptp->dev_clk_regs.phy_clk_ns_h = IDPF_PCI_REG_ADDR(hw,
> +                               temp_offset);
> +     temp_offset = LE32_TO_CPU(clock_offsets.cmd_sync_trigger);
> +     ptp->dev_clk_regs.cmd_sync = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +
> +cross_tstamp:
> +     access_type = (enum idpf_ptp_access)ptp->get_cross_tstamp_access;
> +     if (access_type != IDPF_PTP_DIRECT)
> +             goto discipline_clock;
> +
> +     cross_tstamp_offsets = recv_ptp_caps_msg->cross_time_offsets;
> +
> +     temp_offset = LE32_TO_CPU(cross_tstamp_offsets.sys_time_ns_l);
> +     ptp->dev_clk_regs.sys_time_ns_l = IDPF_PCI_REG_ADDR(hw,
> +                                     temp_offset);

Again, don't wrap. Also, this indentation used neither is a simple
double-indent, nor does it align to brace so please use one style
consistently.

> +     temp_offset = LE32_TO_CPU(cross_tstamp_offsets.sys_time_ns_h);
> +     ptp->dev_clk_regs.sys_time_ns_h = IDPF_PCI_REG_ADDR(hw,
> +                                     temp_offset);
> +     temp_offset = LE32_TO_CPU(cross_tstamp_offsets.cmd_sync_trigger);
> +     ptp->dev_clk_regs.cmd_sync = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +
> +discipline_clock:
> +     access_type = (enum idpf_ptp_access)ptp->adj_dev_clk_time_access;
> +     if (access_type != IDPF_PTP_DIRECT)
> +             return err;
> +
> +     clk_adj_offsets = recv_ptp_caps_msg->clk_adj_offsets;
> +
> +     /* Device clock offsets */
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.dev_clk_cmd_type);
> +     ptp->dev_clk_regs.cmd = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.dev_clk_incval_l);
> +     ptp->dev_clk_regs.incval_l = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.dev_clk_incval_h);
> +     ptp->dev_clk_regs.incval_h = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.dev_clk_shadj_l);
> +     ptp->dev_clk_regs.shadj_l = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.dev_clk_shadj_h);
> +     ptp->dev_clk_regs.shadj_h = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +
> +     /* PHY clock offsets */
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.phy_clk_cmd_type);
> +     ptp->dev_clk_regs.phy_cmd = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.phy_clk_incval_l);
> +     ptp->dev_clk_regs.phy_incval_l = IDPF_PCI_REG_ADDR(hw,
> +                               temp_offset);
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.phy_clk_incval_h);
> +     ptp->dev_clk_regs.phy_incval_h = IDPF_PCI_REG_ADDR(hw,
> +                               temp_offset);
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.phy_clk_shadj_l);
> +     ptp->dev_clk_regs.phy_shadj_l = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +     temp_offset = LE32_TO_CPU(clk_adj_offsets.phy_clk_shadj_h);
> +     ptp->dev_clk_regs.phy_shadj_h = IDPF_PCI_REG_ADDR(hw, temp_offset);
> +
> +     return err;
> +}
> +
> +/**
> + * idpf_ptp_enable_shtime - Enable shadow time and execute a command
> + * @adapter: Driver specific private structure
> + */
> +static void idpf_ptp_enable_shtime(struct idpf_adapter *adapter)
> +{
> +     uint32_t shtime_enable, exec_cmd;
> +
> +     /* Get offsets */
> +     shtime_enable = adapter->ptp->cmd.shtime_enable_mask;
> +     exec_cmd = adapter->ptp->cmd.exec_cmd_mask;
> +
> +     /* Set the shtime en and the sync field */
> +     IDPF_PCI_REG_WRITE(adapter->ptp->dev_clk_regs.cmd_sync, shtime_enable);
> +     IDPF_PCI_REG_WRITE(adapter->ptp->dev_clk_regs.cmd_sync, exec_cmd | 
> shtime_enable);
> +}
> +
> +/**
> + * idpf_ptp_get_dev_clk_time - Send virtchnl get device clk time message
> + * @adapter: Driver specific private structure
> + * @dev_clk_time: Pointer to the device clock structure where the value is 
> set
> + *
> + * Send virtchnl get time message to get the time of the clock.
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +int idpf_ptp_get_dev_clk_time(struct idpf_adapter *adapter,
> +                           struct idpf_ptp_dev_timers *dev_clk_time)
> +{
> +     struct virtchnl2_ptp_get_dev_clk_time get_dev_clk_time_msg = { };
> +     struct idpf_cmd_info args = { };

As above, use {0} as initializer.

> +     int err;
> +     u64 dev_time;
> +
> +     args.ops = VIRTCHNL2_OP_PTP_GET_DEV_CLK_TIME;
> +     args.in_args = (uint8_t *)&get_dev_clk_time_msg;
> +     args.in_args_size = sizeof(get_dev_clk_time_msg);
> +     args.out_buffer = adapter->mbx_resp;
> +     args.out_size = sizeof(get_dev_clk_time_msg);
> +
> +     err = idpf_vc_cmd_execute(adapter, &args);
> +     if (err < 0)
> +             return err;
> +
> +     get_dev_clk_time_msg = *(struct virtchnl2_ptp_get_dev_clk_time 
> *)args.out_buffer;
> +     dev_time = LE64_TO_CPU(get_dev_clk_time_msg.dev_time_ns);
> +     dev_clk_time->dev_clk_time_ns = dev_time;
> +
> +     return err;
> +}
> +
> +/**
> + * idpf_ptp_get_cross_time - Send virtchnl get cross time message
> + * @adapter: Driver specific private structure
> + * @cross_time: Pointer to the device clock structure where the value is set
> + *
> + * Send virtchnl get cross time message to get the time of the clock and the
> + * system time.
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +int idpf_ptp_get_cross_time(struct idpf_adapter *adapter,
> +                         struct idpf_ptp_dev_timers *cross_time)
> +{
> +     struct virtchnl2_ptp_get_cross_time cross_time_msg = { };
> +     struct idpf_cmd_info args = { };
> +     int err;
> +
> +     args.ops = VIRTCHNL2_OP_PTP_GET_CROSS_TIME;
> +     args.in_args = (uint8_t *)&cross_time_msg;
> +     args.in_args_size = sizeof(cross_time_msg);
> +     args.out_buffer = adapter->mbx_resp;
> +     args.out_size = sizeof(cross_time_msg);
> +
> +     err = idpf_vc_cmd_execute(adapter, &args);
> +     if (err < 0)
> +             return err;
> +
> +     cross_time_msg = *(struct virtchnl2_ptp_get_cross_time 
> *)args.out_buffer;
> +     cross_time->dev_clk_time_ns = LE64_TO_CPU(cross_time_msg.dev_time_ns);
> +     cross_time->sys_time_ns = LE64_TO_CPU(cross_time_msg.sys_time_ns);
> +
> +     return err;
> +}
> +
> +/**
> + * idpf_ptp_set_dev_clk_time - Send virtchnl set device time message
> + * @adapter: Driver specific private structure
> + * @time: New time value
> + *
> + * Send virtchnl set time message to set the time of the clock.
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +int idpf_ptp_set_dev_clk_time(struct idpf_adapter *adapter, u64 time)
> +{
> +     struct virtchnl2_ptp_set_dev_clk_time set_dev_clk_time_msg = { };
> +     struct idpf_cmd_info args = { };
> +     int err;
> +
> +     set_dev_clk_time_msg.dev_time_ns = CPU_TO_LE64(time);
> +
> +     args.ops = VIRTCHNL2_OP_PTP_SET_DEV_CLK_TIME;
> +     args.in_args = (uint8_t *)&set_dev_clk_time_msg;
> +     args.in_args_size = sizeof(set_dev_clk_time_msg);
> +     args.out_buffer = adapter->mbx_resp;
> +     args.out_size = sizeof(set_dev_clk_time_msg);
> +
> +     err = idpf_vc_cmd_execute(adapter, &args);
> +     if (err < 0)
> +             return err;
> +
> +     return err;
> +}
> +
> +/**
> + * idpf_ptp_adj_dev_clk_time - Send virtchnl adj device clock time message
> + * @adapter: Driver specific private structure
> + * @delta: Offset in nanoseconds to adjust the time by
> + *
> + * Send virtchnl adj time message to adjust the clock by the indicated delta.
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +int idpf_ptp_adj_dev_clk_time(struct idpf_adapter *adapter, int64_t delta)
> +{
> +     struct virtchnl2_ptp_adj_dev_clk_time adj_dev_clk_time_msg = { };
> +     struct idpf_cmd_info args = { };
> +     int err;
> +
> +     adj_dev_clk_time_msg.delta = CPU_TO_LE64(delta);
> +
> +     args.ops = VIRTCHNL2_OP_PTP_ADJ_DEV_CLK_TIME;
> +     args.in_args = (uint8_t *)&adj_dev_clk_time_msg;
> +     args.in_args_size = sizeof(adj_dev_clk_time_msg);
> +     args.out_buffer = adapter->mbx_resp;
> +     args.out_size = sizeof(adj_dev_clk_time_msg);
> +
> +     err = idpf_vc_cmd_execute(adapter, &args);
> +     if (err < 0)
> +             return err;
> +
> +     return err;
> +}
> +
> +/**
> + * idpf_ptp_adj_dev_clk_fine - Send virtchnl adj time message
> + * @adapter: Driver specific private structure
> + * @incval: Source timer increment value per clock cycle
> + *
> + * Send virtchnl adj fine message to adjust the frequency of the clock by
> + * incval.
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +int idpf_ptp_adj_dev_clk_fine(struct idpf_adapter *adapter, u64 incval)
> +{
> +     struct virtchnl2_ptp_adj_dev_clk_fine adj_dev_clk_fine_msg = { };
> +     struct idpf_cmd_info args = { };
> +     int err;
> +
> +     adj_dev_clk_fine_msg.incval = CPU_TO_LE64(incval);
> +
> +     args.ops = VIRTCHNL2_OP_PTP_ADJ_DEV_CLK_FINE;
> +     args.in_args = (uint8_t *)&adj_dev_clk_fine_msg;
> +     args.in_args_size = sizeof(adj_dev_clk_fine_msg);
> +     args.out_buffer = adapter->mbx_resp;
> +     args.out_size = sizeof(adj_dev_clk_fine_msg);
> +
> +     err = idpf_vc_cmd_execute(adapter, &args);
> +     if (err < 0)
> +             return err;
> +
> +     return err;
> +}
> +
> +/**
> + * idpf_ptp_get_vport_tstamps_caps - Send virtchnl to get tstamps caps for 
> vport
> + * @vport: Virtual port structure
> + *
> + * Send virtchnl get vport tstamps caps message to receive the set of tstamp
> + * capabilities per vport.
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +int idpf_ptp_get_vport_tstamps_caps(struct idpf_vport *vport)
> +{
> +     struct virtchnl2_ptp_get_vport_tx_tstamp_caps send_tx_tstamp_caps = { };
> +     struct virtchnl2_ptp_get_vport_tx_tstamp_caps *rcv_tx_tstamp_caps;
> +     struct virtchnl2_ptp_tx_tstamp_latch_caps tx_tstamp_latch_caps;
> +     enum idpf_ptp_access tstamp_access, get_dev_clk_access;
> +     struct idpf_ptp_vport_tx_tstamp_caps *tstamp_caps;
> +     struct idpf_ptp *ptp = vport->adapter->ptp;
> +     struct idpf_cmd_info args = { };
> +     int err;
> +     u16 num_latches, i;
> +     u32 size;
> +
> +     if (ptp == NULL)
> +             return -EOPNOTSUPP;
> +
> +     tstamp_access = (enum idpf_ptp_access)ptp->tx_tstamp_access;
> +     get_dev_clk_access = (enum idpf_ptp_access)ptp->get_dev_clk_time_access;
> +     if (tstamp_access == IDPF_PTP_NONE ||
> +         get_dev_clk_access == IDPF_PTP_NONE)
> +             return -EOPNOTSUPP;
> +
> +     send_tx_tstamp_caps.vport_id = CPU_TO_LE32(vport->vport_id);
> +
> +     args.ops = VIRTCHNL2_OP_PTP_GET_VPORT_TX_TSTAMP_CAPS;
> +     args.in_args = (uint8_t *)&send_tx_tstamp_caps;
> +     args.in_args_size = sizeof(send_tx_tstamp_caps);
> +     args.out_size = IDPF_CTLQ_MAX_BUF_LEN;
> +     args.out_buffer = vport->adapter->mbx_resp;
> +
> +     err = idpf_vc_cmd_execute(vport->adapter, &args);
> +     if (err < 0)
> +             return err;
> +
> +     rcv_tx_tstamp_caps = (struct virtchnl2_ptp_get_vport_tx_tstamp_caps *)
> +                          args.out_buffer;
> +     num_latches = LE16_TO_CPU(rcv_tx_tstamp_caps->num_latches);
> +     size = sizeof(struct idpf_ptp_vport_tx_tstamp_caps) +
> +            sizeof(struct idpf_ptp_tx_tstamp) * num_latches;
> +     tstamp_caps = rte_zmalloc(NULL, size, 0);
> +     if (tstamp_caps == NULL)
> +             return -ENOMEM;
> +
> +     tstamp_caps->access = true;
> +     tstamp_caps->num_entries = num_latches;
> +
> +     tstamp_caps->tstamp_ns_lo_bit = rcv_tx_tstamp_caps->tstamp_ns_lo_bit;
> +
> +     for (i = 0; i < tstamp_caps->num_entries; i++) {
> +             __le32 offset_l, offset_h;
> +
> +             tx_tstamp_latch_caps = rcv_tx_tstamp_caps->tstamp_latches[i];
> +
> +             if (tstamp_access == IDPF_PTP_DIRECT) {
> +                     offset_l = tx_tstamp_latch_caps.tx_latch_reg_offset_l;
> +                     offset_h = tx_tstamp_latch_caps.tx_latch_reg_offset_h;
> +                     tstamp_caps->tx_tstamp[i].tx_latch_reg_offset_l = 
> LE32_TO_CPU(offset_l);
> +                     tstamp_caps->tx_tstamp[i].tx_latch_reg_offset_h = 
> LE32_TO_CPU(offset_h);
> +             }
> +             tstamp_caps->tx_tstamp[i].idx = tx_tstamp_latch_caps.index;
> +     }
> +
> +     tstamp_caps->latched_idx = -1;
> +     vport->tx_tstamp_caps = tstamp_caps;
> +
> +     return err;
> +}
> +
> +/**
> + * idpf_ptp_get_tstamp_value - Get the Tx timestamp value and provide it
> + *                          back to the skb.
> + * @vport: Virtual port structure
> + * @tstamp_latch: Tx timestamp latch structure fulfilled by the Control Plane
> + * @tx_tstamp: Tx timestamp structure to be fulfilled with the timestamp 
> value
> + *
> + * Read the value of the Tx timestamp for a given latch received from the
> + * Control Plane.
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +static int
> +idpf_ptp_get_tstamp_value(struct idpf_vport *vport,
> +                       struct virtchnl2_ptp_tx_tstamp_latch *tstamp_latch,
> +                       struct idpf_ptp_tx_tstamp *tx_tstamp)
> +{
> +     struct idpf_ptp_vport_tx_tstamp_caps *tx_tstamp_caps;
> +     u8 tstamp_ns_lo_bit;
> +
> +     tx_tstamp_caps = vport->tx_tstamp_caps;
> +     tstamp_ns_lo_bit = tx_tstamp_caps->tstamp_ns_lo_bit;
> +
> +     tx_tstamp->tstamp = LE64_TO_CPU(tstamp_latch->tstamp);
> +     tx_tstamp->tstamp >>= tstamp_ns_lo_bit;
> +
> +     return 0;
> +}
> +
> +/**
> + * idpf_ptp_get_tx_tstamp - Send virtchnl get Tx timestamp latches message
> + * @vport: Virtual port structure
> + *
> + * Send virtchnl get Tx tstamp message to read the value of the HW timestamp.
> + * The message contains a list of indexes set in the Tx descriptors.
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +int idpf_ptp_get_tx_tstamp(struct idpf_vport *vport)
> +{
> +     struct virtchnl2_ptp_get_vport_tx_tstamp_latches *send_tx_tstamp_msg;
> +     struct virtchnl2_ptp_get_vport_tx_tstamp_latches *recv_tx_tstamp_msg;
> +     struct idpf_ptp_vport_tx_tstamp_caps *tx_tstamp_caps;
> +     struct virtchnl2_ptp_tx_tstamp_latch tstamp_latch;
> +     struct idpf_ptp_tx_tstamp *ptp_tx_tstamp;
> +     struct idpf_cmd_info args = { };
> +     int size, msg_size;
> +     u32 vport_id;
> +     u16 num_latches, id;
> +     int err;
> +
> +     tx_tstamp_caps = vport->tx_tstamp_caps;
> +     ptp_tx_tstamp = tx_tstamp_caps->tx_tstamp;
> +
> +     size = sizeof(struct virtchnl2_ptp_get_vport_tx_tstamp_latches) +
> +           sizeof(struct virtchnl2_ptp_tx_tstamp_latch) *
> +           tx_tstamp_caps->num_entries;
> +     send_tx_tstamp_msg = rte_zmalloc(NULL, size, 0);
> +     if (send_tx_tstamp_msg == NULL)
> +             return -ENOMEM;
> +
> +     for (id = 0; id < tx_tstamp_caps->num_entries; id++,
> +             ptp_tx_tstamp++)
> +             send_tx_tstamp_msg->tstamp_latches[id].index =
> +                                                                             
> ptp_tx_tstamp->idx;

Identation in this loop badly needs to be reworked.

> +     send_tx_tstamp_msg->get_devtime_with_txtstmp = 1;
> +
> +     msg_size = sizeof(struct virtchnl2_ptp_get_vport_tx_tstamp_latches) +
> +                sizeof(struct virtchnl2_ptp_tx_tstamp_latch) * id;
> +     send_tx_tstamp_msg->vport_id = CPU_TO_LE32(vport->vport_id);
> +     send_tx_tstamp_msg->num_latches = CPU_TO_LE16(id);
> +
> +     args.ops = VIRTCHNL2_OP_PTP_GET_VPORT_TX_TSTAMP;
> +     args.in_args = (uint8_t *)send_tx_tstamp_msg;
> +     args.in_args_size = msg_size;
> +     args.out_size = msg_size;
> +     args.out_buffer = vport->adapter->mbx_resp;
> +
> +     err = idpf_vc_cmd_execute(vport->adapter, &args);
> +     rte_free(send_tx_tstamp_msg);
> +     if (err < 0)
> +             return err;
> +
> +     recv_tx_tstamp_msg = (struct virtchnl2_ptp_get_vport_tx_tstamp_latches 
> *)
> +                          args.out_buffer;
> +     vport_id = LE32_TO_CPU(recv_tx_tstamp_msg->vport_id);
> +     if (vport->vport_id != vport_id)
> +             return -EINVAL;
> +
> +     num_latches = LE16_TO_CPU(recv_tx_tstamp_msg->num_latches);
> +
> +     ptp_tx_tstamp = tx_tstamp_caps->tx_tstamp;
> +     for (id = 0; id < num_latches; id++, ptp_tx_tstamp++) {
> +             tstamp_latch = recv_tx_tstamp_msg->tstamp_latches[id];
> +
> +             if (!tstamp_latch.valid)
> +                     continue;
> +
> +             err = idpf_ptp_get_tstamp_value(vport, &tstamp_latch,
> +                                             ptp_tx_tstamp);
> +             if (err == 0) {
> +                     tx_tstamp_caps->latched_idx = id;
> +                     vport->adapter->time_hw = 
> recv_tx_tstamp_msg->device_time;
> +             }
> +             break;
> +     }
> +     return err;
> +}
> +
> +/**
> + * idpf_ptp_read_src_clk_reg_direct - Read directly the main timer value
> + * @adapter: Driver specific private structure
> + *
> + * Return: the device clock time.
> + */
> +static u64 idpf_ptp_read_src_clk_reg_direct(struct idpf_adapter *adapter)
> +{
> +     struct idpf_ptp *ptp = adapter->ptp;
> +     u32 hi, lo;
> +
> +     idpf_ptp_enable_shtime(adapter);
> +
> +     lo = IDPF_PCI_REG(ptp->dev_clk_regs.dev_clk_ns_l);
> +     hi = IDPF_PCI_REG(ptp->dev_clk_regs.dev_clk_ns_h);
> +
> +     return ((u64)hi << 32) | lo;
> +}
> +
> +/**
> + * idpf_ptp_read_src_clk_reg_mailbox - Read the main timer value through 
> mailbox
> + * @adapter: Driver specific private structure
> + * @src_clk: Returned main timer value in nanoseconds unit
> + *
> + * Return: 0 on success, -errno otherwise.
> + */
> +static int idpf_ptp_read_src_clk_reg_mailbox(struct idpf_adapter *adapter,
> +                                          u64 *src_clk)
> +{
> +     struct idpf_ptp_dev_timers clk_time;
> +     int err;
> +
> +     err = idpf_ptp_get_dev_clk_time(adapter, &clk_time);
> +     if (err)
> +             return err;
> +
> +     *src_clk = clk_time.dev_clk_time_ns;
> +
> +     return 0;
> +}
> +
> +/**
> + * idpf_ptp_read_src_clk_reg - Read the main timer value
> + * @adapter: Driver specific private structure
> + * @src_clk: Returned main timer value in nanoseconds unit
> + *
> + * Return: the device clock time on success, -errno otherwise.
> + */
> +int idpf_ptp_read_src_clk_reg(struct idpf_adapter *adapter, u64 *src_clk)
> +{
> +     if (adapter->ptp == NULL)
> +             return 0;

Should this not be an error, returning -EINVAL?

> +     switch ((enum idpf_ptp_access)adapter->ptp->get_dev_clk_time_access) {
> +     case IDPF_PTP_NONE:
> +             return -EOPNOTSUPP;

Strictly speaking, you don't need this, it's handled by the default case.

> +     case IDPF_PTP_MAILBOX:
> +             return idpf_ptp_read_src_clk_reg_mailbox(adapter, src_clk);
> +     case IDPF_PTP_DIRECT:
> +             *src_clk = idpf_ptp_read_src_clk_reg_direct(adapter);
> +             break;

Why have the function types for direct and mailbox different? Why not have
both just take src_clk as parameter and return 0 on success, so you can
have similar use of them. It would also possibly allow function pointer use
in future, if they are the same types.

> +     default:
> +             return -EOPNOTSUPP;
> +     }
> +
> +     return 0;
> +}
> diff --git a/drivers/net/intel/idpf/idpf_ptp.h 
> b/drivers/net/intel/idpf/idpf_ptp.h
> new file mode 100644
> index 0000000000..68ac66ae91
> --- /dev/null
> +++ b/drivers/net/intel/idpf/idpf_ptp.h
> @@ -0,0 +1,227 @@
> +/* SPDX-License-Identifier: BSD-3-Clause
> + * Copyright(c) 2025 Intel Corporation
> + */
> +
> +#ifndef _IDPF_PTP_H_
> +#define _IDPF_PTP_H_
> +
> +#include "idpf_osdep.h"
> +#include <rte_time.h>
> +#include "idpf_common_device.h"
> +
> +/**
> + * @struct idpf_ptp_cmd - PTP command masks
> + * @exec_cmd_mask: mask to trigger command execution
> + * @shtime_enable_mask: mask to enable shadow time
> + */
> +struct idpf_ptp_cmd {
> +     uint32_t exec_cmd_mask;
> +     uint32_t shtime_enable_mask;
> +};
> +
> +/* @struct idpf_ptp_dev_clk_regs - PTP device registers
> + * @dev_clk_ns_l: low part of the device clock register
> + * @dev_clk_ns_h: high part of the device clock register
> + * @phy_clk_ns_l: low part of the PHY clock register
> + * @phy_clk_ns_h: high part of the PHY clock register
> + * @sys_time_ns_l: low part of the system time register
> + * @sys_time_ns_h: high part of the system time register
> + * @incval_l: low part of the increment value register
> + * @incval_h: high part of the increment value register
> + * @shadj_l: low part of the shadow adjust register
> + * @shadj_h: high part of the shadow adjust register
> + * phy_incval_l: low part of the PHY increment value register
> + * phy_incval_h: high part of the PHY increment value register
> + * phy_shadj_l: low part of the PHY shadow adjust register
> + * phy_shadj_h: high part of the PHY shadow adjust register
> + * @cmd: PTP command register
> + * @phy_cmd: PHY command register
> + * @cmd_sync: PTP command synchronization register
> + */
> +struct idpf_ptp_dev_clk_regs {
> +     /* Main clock */
> +     volatile uint32_t *dev_clk_ns_l;
> +     volatile uint32_t *dev_clk_ns_h;
> +
> +     /* PHY timer */
> +     volatile uint32_t *phy_clk_ns_l;
> +     volatile uint32_t *phy_clk_ns_h;
> +
> +     /* System time */
> +     volatile uint32_t *sys_time_ns_l;
> +     volatile uint32_t *sys_time_ns_h;
> +
> +     /* Main timer adjustments */
> +     volatile uint32_t *incval_l;
> +     volatile uint32_t *incval_h;
> +     volatile uint32_t *shadj_l;
> +     volatile uint32_t *shadj_h;
> +
> +     /* PHY timer adjustments */
> +     volatile uint32_t *phy_incval_l;
> +     volatile uint32_t *phy_incval_h;
> +     volatile uint32_t *phy_shadj_l;
> +     volatile uint32_t *phy_shadj_h;
> +
> +     /* Command */
> +     volatile uint32_t *cmd;
> +     volatile uint32_t *phy_cmd;
> +     volatile uint32_t *cmd_sync;
> +};
> +
> +/**
> + * @enum idpf_ptp_access - the type of access to PTP operations
> + * @IDPF_PTP_NONE: no access
> + * @IDPF_PTP_DIRECT: direct access through BAR registers
> + * @IDPF_PTP_MAILBOX: access through mailbox messages
> + */
> +enum idpf_ptp_access {
> +     IDPF_PTP_NONE = 0,
> +     IDPF_PTP_DIRECT,
> +     IDPF_PTP_MAILBOX,
> +};
> +
> +/**
> + * @struct idpf_ptp_secondary_mbx - PTP secondary mailbox
> + * @peer_mbx_q_id: PTP mailbox queue ID
> + * @peer_id: Peer ID for PTP Device Control daemon
> + * @valid: indicates whether secondary mailblox is supported by the Control
> + *      Plane
> + */
> +struct idpf_ptp_secondary_mbx {
> +     uint16_t peer_mbx_q_id;
> +     uint16_t peer_id;
> +     bool valid:1;
> +};
> +
> +/**
> + * @enum idpf_ptp_tx_tstamp_state - Tx timestamp states
> + * @IDPF_PTP_FREE: Tx timestamp index free to use
> + * @IDPF_PTP_REQUEST: Tx timestamp index set to the Tx descriptor
> + * @IDPF_PTP_READ_VALUE: Tx timestamp value ready to be read
> + */
> +enum idpf_ptp_tx_tstamp_state {
> +     IDPF_PTP_FREE,
> +     IDPF_PTP_REQUEST,
> +     IDPF_PTP_READ_VALUE,
> +};
> +
> +/**
> + * @struct idpf_ptp_tx_tstamp - Parameters for Tx timestamping
> + * @list_member: the list member structure
> + * @tx_latch_reg_offset_l: Tx tstamp latch low register offset
> + * @tx_latch_reg_offset_h: Tx tstamp latch high register offset
> + * @tstamp: the Tx tstamp value
> + * @idx: the index of the Tx tstamp
> + */
> +struct idpf_ptp_tx_tstamp {
> +     uint64_t tstamp;
> +     uint32_t tx_latch_reg_offset_l;
> +     uint32_t tx_latch_reg_offset_h;
> +     uint32_t idx;
> +};
> +
> +/**
> + * @struct idpf_ptp_vport_tx_tstamp_caps - Tx timestamp capabilities
> + * @vport_id: the vport id
> + * @num_entries: the number of negotiated Tx timestamp entries
> + * @tstamp_ns_lo_bit: first bit for nanosecond part of the timestamp
> + * @access: indicates an access to Tx timestamp
> + * @latches_index: the index  of the latched Tx timestamps
> + * @tx_tstamp: array of Tx timestamp parameters
> + */
> +struct idpf_ptp_vport_tx_tstamp_caps {
> +     uint32_t vport_id;
> +     uint16_t num_entries;
> +     uint16_t tstamp_ns_lo_bit;
> +     uint16_t latched_idx;
> +     bool access:1;
> +     struct idpf_ptp_tx_tstamp tx_tstamp[];
> +};
> +
> +/**
> + * @struct idpf_ptp - PTP parameters
> + * @base_incval: base increment value of the PTP clock
> + * @max_adj: maximum adjustment of the PTP clock
> + * @cmd: HW specific command masks
> + * @dev_clk_regs: the set of registers to access the device clock
> + * @caps: PTP capabilities negotiated with the Control Plane
> + * @get_dev_clk_time_access: access type for getting the device clock time
> + * @get_cross_tstamp_access: access type for the cross timestamping
> + * @set_dev_clk_time_access: access type for setting the device clock time
> + * @adj_dev_clk_time_access: access type for the adjusting the device clock
> + * @tx_tstamp_access: access type for the Tx timestamp value read
> + * @rsv: Reserved fields
> + * @secondary_mbx: parameters for using dedicated PTP mailbox
> + */
> +struct idpf_ptp {
> +     uint64_t base_incval;
> +     uint64_t max_adj;
> +     struct idpf_ptp_cmd cmd;
> +     struct idpf_ptp_dev_clk_regs dev_clk_regs;
> +     uint32_t caps;
> +     uint8_t get_dev_clk_time_access:2;
> +     uint8_t get_cross_tstamp_access:2;
> +     uint8_t set_dev_clk_time_access:2;
> +     uint8_t adj_dev_clk_time_access:2;
> +     uint8_t tx_tstamp_access:2;
> +     uint8_t rsv:6;
> +     struct idpf_ptp_secondary_mbx secondary_mbx;
> +};
> +
> +/**
> + * @struct idpf_ptp_dev_timers - System time and device time values
> + * @sys_time_ns: system time value expressed in nanoseconds
> + * @dev_clk_time_ns: device clock time value expressed in nanoseconds
> + */
> +struct idpf_ptp_dev_timers {
> +     uint64_t sys_time_ns;
> +     uint64_t dev_clk_time_ns;
> +};
> +
> +int idpf_ptp_get_caps(struct idpf_adapter *adapter);
> +int idpf_ptp_read_src_clk_reg(struct idpf_adapter *adapter, uint64_t 
> *src_clk);
> +int idpf_ptp_get_dev_clk_time(struct idpf_adapter *adapter,
> +                           struct idpf_ptp_dev_timers *dev_clk_time);
> +int idpf_ptp_get_cross_time(struct idpf_adapter *adapter,
> +                         struct idpf_ptp_dev_timers *cross_time);
> +int idpf_ptp_set_dev_clk_time(struct idpf_adapter *adapter, uint64_t time);
> +int idpf_ptp_adj_dev_clk_fine(struct idpf_adapter *adapter, uint64_t incval);
> +int idpf_ptp_adj_dev_clk_time(struct idpf_adapter *adapter, int64_t delta);
> +int idpf_ptp_get_vport_tstamps_caps(struct idpf_vport *vport);
> +int idpf_ptp_get_tx_tstamp(struct idpf_vport *vport);
> +
> +/* Helper function to convert a 32b nanoseconds timestamp to 64b. */
> +static inline uint64_t
> +idpf_tstamp_convert_32b_64b(struct idpf_adapter *ad, uint32_t flag,
> +                         bool is_rx, uint32_t in_timestamp)
> +{
> +     const uint64_t mask = 0xFFFFFFFF;
> +     uint32_t phc_time_lo, delta;
> +     uint64_t ns;
> +
> +     if (flag != 0)
> +             idpf_ptp_read_src_clk_reg(ad, &ad->time_hw);
> +
> +    /* Extract the lower 32 bits of the PHC time */
> +     phc_time_lo = (uint32_t)(ad->time_hw);
> +
> +     /* Calculate the delta between the lower 32bits of the cached PHC
> +      * time and the in_timestamp value.
> +      */
> +     delta = in_timestamp - phc_time_lo;
> +
> +     if (delta > mask / 2) {
> +             /* Reverse the delta calculation here */
> +             delta = phc_time_lo - in_timestamp;
> +             ns = ad->time_hw - delta;
> +     } else {
> +             if (is_rx)
> +                     ns = ad->time_hw - delta;
> +             else
> +                     ns = ad->time_hw + delta;
> +     }
> +
> +     return ns;
> +}
> +#endif /* _IDPF_PTP_H_ */
> diff --git a/drivers/net/intel/idpf/meson.build 
> b/drivers/net/intel/idpf/meson.build
> index a805d02ea2..5a4a3c2259 100644
> --- a/drivers/net/intel/idpf/meson.build
> +++ b/drivers/net/intel/idpf/meson.build
> @@ -18,6 +18,7 @@ sources += files(
>  
>          'idpf_ethdev.c',
>          'idpf_rxtx.c',
> +        'idpf_ptp.c',

All lists in DPDK should be in alphabetical order, unless there is a reason
they can't be. Therefore, this new C file should be above idpf_rxtx.c in
the list.

>  )
>  
>  if arch_subdir == 'x86' and dpdk_conf.get('RTE_IOVA_IN_MBUF') == 1
> -- 
> 2.47.1
> 

Reply via email to