Commit c49bba01 authored by David S. Miller's avatar David S. Miller

Merge branch 'ptp-multiple-readers'

Xabier Marquiegui says:

====================
ptp: Support for multiple filtered timestamp event queue readers

On systems with multiple timestamp event channels, there can be scenarios
where multiple userspace readers want to access the timestamping data for
various purposes.

One such example is wanting to use a pps out for time synchronization, and
wanting to timestamp external events with the synchronized time base
simultaneously.

Timestmp event consumers on the other hand, are often interested in a
subset of the available timestamp channels. linuxptp ts2phc, for example,
is not happy if more than one timestamping channel is active on the device
it is reading from.

Linked lists are introduced to support multiple timestamp event queue
consumers, and timestamp event channel filters through IOCTLs, as well as
a debugfs interface to do some simple verifications.

Xabier Marquiegui (6):
  posix-clock: introduce posix_clock_context concept
  ptp: Replace timestamp event queue with linked list
  ptp: support multiple timestamp event readers
  ptp: support event queue reader channel masks
  ptp: add debugfs interface to see applied channel masks
  ptp: add testptp mask test

 drivers/ptp/ptp_chardev.c                   | 129 ++++++++++++++++----
 drivers/ptp/ptp_clock.c                     |  45 ++++++-
 drivers/ptp/ptp_private.h                   |  28 +++--
 drivers/ptp/ptp_sysfs.c                     |  13 +-
 include/linux/posix-clock.h                 |  35 ++++--
 include/uapi/linux/ptp_clock.h              |   2 +
 kernel/time/posix-clock.c                   |  36 ++++--
 tools/testing/selftests/ptp/ptpchmaskfmt.sh |  14 +++
 tools/testing/selftests/ptp/testptp.c       |  19 ++-
 9 files changed, 261 insertions(+), 60 deletions(-)
 create mode 100644 tools/testing/selftests/ptp/ptpchmaskfmt.sh

---
v6:
  - correct commit message
  - correct coding style
v5: https://lore.kernel.org/netdev/cover.1696804243.git.reibax@gmail.com/
  - fix spelling on commit message
  - fix memory leak on ptp_open
v4: https://lore.kernel.org/netdev/cover.1696511486.git.reibax@gmail.com/
  - split modifications in different patches for improved organization
  - rename posix_clock_user to posix_clock_context
  - remove unnecessary flush_users clock operation
  - remove unnecessary tests
  - simpler queue clean procedure
  - fix/clean comment lines
  - simplified release procedures
  - filter modifications exclusive to currently open instance for
    simplicity and security
  - expand mask to 2048 channels
  - make more secure and simple: mask is only applied to the testptp
    instance. Use debugfs to verify effects.
v3: https://lore.kernel.org/netdev/20230928133544.3642650-1-reibax@gmail.com/
  - add this patchset overview file
  - fix use of safe and non safe linked lists for loops
  - introduce new posix_clock private_data and ida object ids for better
    dicrimination of timestamp consumers
  - safer resource release procedures
  - filter application by object id, aided by process id
  - friendlier testptp implementation of event queue channel filters
v2: https://lore.kernel.org/netdev/20230912220217.2008895-1-reibax@gmail.com/
  - fix ptp_poll() return value
  - Style changes to comform to checkpatch strict suggestions
  - more coherent ptp_read error exit routines
  - fix testptp compilation error: unknown type name 'pid_t'
  - rename mask variable for easier code traceability
  - more detailed commit message with two examples
v1: https://lore.kernel.org/netdev/20230906104754.1324412-2-reibax@gmail.com/
====================
Signed-off-by: default avatarXabier Marquiegui <reibax@gmail.com>
Suggested-by: default avatarRichard Cochran <richardcochran@gmail.com>
Suggested-by: default avatarVinicius Costa Gomes <vinicius.gomes@intel.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 99620ea0 26285e68
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
#include <linux/sched.h> #include <linux/sched.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/timekeeping.h> #include <linux/timekeeping.h>
#include <linux/debugfs.h>
#include <linux/nospec.h> #include <linux/nospec.h>
...@@ -101,19 +102,67 @@ int ptp_set_pinfunc(struct ptp_clock *ptp, unsigned int pin, ...@@ -101,19 +102,67 @@ int ptp_set_pinfunc(struct ptp_clock *ptp, unsigned int pin,
return 0; return 0;
} }
int ptp_open(struct posix_clock *pc, fmode_t fmode) int ptp_open(struct posix_clock_context *pccontext, fmode_t fmode)
{ {
struct ptp_clock *ptp =
container_of(pccontext->clk, struct ptp_clock, clock);
struct timestamp_event_queue *queue;
char debugfsname[32];
queue = kzalloc(sizeof(*queue), GFP_KERNEL);
if (!queue)
return -EINVAL;
queue->mask = bitmap_alloc(PTP_MAX_CHANNELS, GFP_KERNEL);
if (!queue->mask) {
kfree(queue);
return -EINVAL;
}
bitmap_set(queue->mask, 0, PTP_MAX_CHANNELS);
spin_lock_init(&queue->lock);
list_add_tail(&queue->qlist, &ptp->tsevqs);
pccontext->private_clkdata = queue;
/* Debugfs contents */
sprintf(debugfsname, "0x%p", queue);
queue->debugfs_instance =
debugfs_create_dir(debugfsname, ptp->debugfs_root);
queue->dfs_bitmap.array = (u32 *)queue->mask;
queue->dfs_bitmap.n_elements =
DIV_ROUND_UP(PTP_MAX_CHANNELS, BITS_PER_BYTE * sizeof(u32));
debugfs_create_u32_array("mask", 0444, queue->debugfs_instance,
&queue->dfs_bitmap);
return 0;
}
int ptp_release(struct posix_clock_context *pccontext)
{
struct timestamp_event_queue *queue = pccontext->private_clkdata;
unsigned long flags;
if (queue) {
debugfs_remove(queue->debugfs_instance);
pccontext->private_clkdata = NULL;
spin_lock_irqsave(&queue->lock, flags);
list_del(&queue->qlist);
spin_unlock_irqrestore(&queue->lock, flags);
bitmap_free(queue->mask);
kfree(queue);
}
return 0; return 0;
} }
long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg) long ptp_ioctl(struct posix_clock_context *pccontext, unsigned int cmd,
unsigned long arg)
{ {
struct ptp_clock *ptp = container_of(pc, struct ptp_clock, clock); struct ptp_clock *ptp =
container_of(pccontext->clk, struct ptp_clock, clock);
struct ptp_sys_offset_extended *extoff = NULL; struct ptp_sys_offset_extended *extoff = NULL;
struct ptp_sys_offset_precise precise_offset; struct ptp_sys_offset_precise precise_offset;
struct system_device_crosststamp xtstamp; struct system_device_crosststamp xtstamp;
struct ptp_clock_info *ops = ptp->info; struct ptp_clock_info *ops = ptp->info;
struct ptp_sys_offset *sysoff = NULL; struct ptp_sys_offset *sysoff = NULL;
struct timestamp_event_queue *tsevq;
struct ptp_system_timestamp sts; struct ptp_system_timestamp sts;
struct ptp_clock_request req; struct ptp_clock_request req;
struct ptp_clock_caps caps; struct ptp_clock_caps caps;
...@@ -123,6 +172,8 @@ long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg) ...@@ -123,6 +172,8 @@ long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg)
struct timespec64 ts; struct timespec64 ts;
int enable, err = 0; int enable, err = 0;
tsevq = pccontext->private_clkdata;
switch (cmd) { switch (cmd) {
case PTP_CLOCK_GETCAPS: case PTP_CLOCK_GETCAPS:
...@@ -421,6 +472,22 @@ long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg) ...@@ -421,6 +472,22 @@ long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg)
mutex_unlock(&ptp->pincfg_mux); mutex_unlock(&ptp->pincfg_mux);
break; break;
case PTP_MASK_CLEAR_ALL:
bitmap_clear(tsevq->mask, 0, PTP_MAX_CHANNELS);
break;
case PTP_MASK_EN_SINGLE:
if (copy_from_user(&i, (void __user *)arg, sizeof(i))) {
err = -EFAULT;
break;
}
if (i >= PTP_MAX_CHANNELS) {
err = -EFAULT;
break;
}
set_bit(i, tsevq->mask);
break;
default: default:
err = -ENOTTY; err = -ENOTTY;
break; break;
...@@ -432,53 +499,65 @@ long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg) ...@@ -432,53 +499,65 @@ long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg)
return err; return err;
} }
__poll_t ptp_poll(struct posix_clock *pc, struct file *fp, poll_table *wait) __poll_t ptp_poll(struct posix_clock_context *pccontext, struct file *fp,
poll_table *wait)
{ {
struct ptp_clock *ptp = container_of(pc, struct ptp_clock, clock); struct ptp_clock *ptp =
container_of(pccontext->clk, struct ptp_clock, clock);
struct timestamp_event_queue *queue;
queue = pccontext->private_clkdata;
if (!queue)
return EPOLLERR;
poll_wait(fp, &ptp->tsev_wq, wait); poll_wait(fp, &ptp->tsev_wq, wait);
return queue_cnt(&ptp->tsevq) ? EPOLLIN : 0; return queue_cnt(queue) ? EPOLLIN : 0;
} }
#define EXTTS_BUFSIZE (PTP_BUF_TIMESTAMPS * sizeof(struct ptp_extts_event)) #define EXTTS_BUFSIZE (PTP_BUF_TIMESTAMPS * sizeof(struct ptp_extts_event))
ssize_t ptp_read(struct posix_clock *pc, ssize_t ptp_read(struct posix_clock_context *pccontext, uint rdflags,
uint rdflags, char __user *buf, size_t cnt) char __user *buf, size_t cnt)
{ {
struct ptp_clock *ptp = container_of(pc, struct ptp_clock, clock); struct ptp_clock *ptp =
struct timestamp_event_queue *queue = &ptp->tsevq; container_of(pccontext->clk, struct ptp_clock, clock);
struct timestamp_event_queue *queue;
struct ptp_extts_event *event; struct ptp_extts_event *event;
unsigned long flags; unsigned long flags;
size_t qcnt, i; size_t qcnt, i;
int result; int result;
if (cnt % sizeof(struct ptp_extts_event) != 0) queue = pccontext->private_clkdata;
return -EINVAL; if (!queue) {
result = -EINVAL;
goto exit;
}
if (cnt % sizeof(struct ptp_extts_event) != 0) {
result = -EINVAL;
goto exit;
}
if (cnt > EXTTS_BUFSIZE) if (cnt > EXTTS_BUFSIZE)
cnt = EXTTS_BUFSIZE; cnt = EXTTS_BUFSIZE;
cnt = cnt / sizeof(struct ptp_extts_event); cnt = cnt / sizeof(struct ptp_extts_event);
if (mutex_lock_interruptible(&ptp->tsevq_mux))
return -ERESTARTSYS;
if (wait_event_interruptible(ptp->tsev_wq, if (wait_event_interruptible(ptp->tsev_wq,
ptp->defunct || queue_cnt(queue))) { ptp->defunct || queue_cnt(queue))) {
mutex_unlock(&ptp->tsevq_mux);
return -ERESTARTSYS; return -ERESTARTSYS;
} }
if (ptp->defunct) { if (ptp->defunct) {
mutex_unlock(&ptp->tsevq_mux); result = -ENODEV;
return -ENODEV; goto exit;
} }
event = kmalloc(EXTTS_BUFSIZE, GFP_KERNEL); event = kmalloc(EXTTS_BUFSIZE, GFP_KERNEL);
if (!event) { if (!event) {
mutex_unlock(&ptp->tsevq_mux); result = -ENOMEM;
return -ENOMEM; goto exit;
} }
spin_lock_irqsave(&queue->lock, flags); spin_lock_irqsave(&queue->lock, flags);
...@@ -497,12 +576,16 @@ ssize_t ptp_read(struct posix_clock *pc, ...@@ -497,12 +576,16 @@ ssize_t ptp_read(struct posix_clock *pc,
cnt = cnt * sizeof(struct ptp_extts_event); cnt = cnt * sizeof(struct ptp_extts_event);
mutex_unlock(&ptp->tsevq_mux);
result = cnt; result = cnt;
if (copy_to_user(buf, event, cnt)) if (copy_to_user(buf, event, cnt)) {
result = -EFAULT; result = -EFAULT;
goto free_event;
}
free_event:
kfree(event); kfree(event);
exit:
if (result < 0)
ptp_release(pccontext);
return result; return result;
} }
...@@ -15,6 +15,7 @@ ...@@ -15,6 +15,7 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/syscalls.h> #include <linux/syscalls.h>
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <linux/debugfs.h>
#include <uapi/linux/sched/types.h> #include <uapi/linux/sched/types.h>
#include "ptp_private.h" #include "ptp_private.h"
...@@ -162,6 +163,7 @@ static struct posix_clock_operations ptp_clock_ops = { ...@@ -162,6 +163,7 @@ static struct posix_clock_operations ptp_clock_ops = {
.clock_settime = ptp_clock_settime, .clock_settime = ptp_clock_settime,
.ioctl = ptp_ioctl, .ioctl = ptp_ioctl,
.open = ptp_open, .open = ptp_open,
.release = ptp_release,
.poll = ptp_poll, .poll = ptp_poll,
.read = ptp_read, .read = ptp_read,
}; };
...@@ -169,12 +171,22 @@ static struct posix_clock_operations ptp_clock_ops = { ...@@ -169,12 +171,22 @@ static struct posix_clock_operations ptp_clock_ops = {
static void ptp_clock_release(struct device *dev) static void ptp_clock_release(struct device *dev)
{ {
struct ptp_clock *ptp = container_of(dev, struct ptp_clock, dev); struct ptp_clock *ptp = container_of(dev, struct ptp_clock, dev);
struct timestamp_event_queue *tsevq;
unsigned long flags;
ptp_cleanup_pin_groups(ptp); ptp_cleanup_pin_groups(ptp);
kfree(ptp->vclock_index); kfree(ptp->vclock_index);
mutex_destroy(&ptp->tsevq_mux);
mutex_destroy(&ptp->pincfg_mux); mutex_destroy(&ptp->pincfg_mux);
mutex_destroy(&ptp->n_vclocks_mux); mutex_destroy(&ptp->n_vclocks_mux);
/* Delete first entry */
tsevq = list_first_entry(&ptp->tsevqs, struct timestamp_event_queue,
qlist);
spin_lock_irqsave(&tsevq->lock, flags);
list_del(&tsevq->qlist);
spin_unlock_irqrestore(&tsevq->lock, flags);
bitmap_free(tsevq->mask);
kfree(tsevq);
debugfs_remove(ptp->debugfs_root);
ida_free(&ptp_clocks_map, ptp->index); ida_free(&ptp_clocks_map, ptp->index);
kfree(ptp); kfree(ptp);
} }
...@@ -206,7 +218,9 @@ struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info, ...@@ -206,7 +218,9 @@ struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info,
struct device *parent) struct device *parent)
{ {
struct ptp_clock *ptp; struct ptp_clock *ptp;
struct timestamp_event_queue *queue = NULL;
int err = 0, index, major = MAJOR(ptp_devt); int err = 0, index, major = MAJOR(ptp_devt);
char debugfsname[8];
size_t size; size_t size;
if (info->n_alarm > PTP_MAX_ALARMS) if (info->n_alarm > PTP_MAX_ALARMS)
...@@ -228,8 +242,16 @@ struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info, ...@@ -228,8 +242,16 @@ struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info,
ptp->info = info; ptp->info = info;
ptp->devid = MKDEV(major, index); ptp->devid = MKDEV(major, index);
ptp->index = index; ptp->index = index;
spin_lock_init(&ptp->tsevq.lock); INIT_LIST_HEAD(&ptp->tsevqs);
mutex_init(&ptp->tsevq_mux); queue = kzalloc(sizeof(*queue), GFP_KERNEL);
if (!queue)
goto no_memory_queue;
list_add_tail(&queue->qlist, &ptp->tsevqs);
queue->mask = bitmap_alloc(PTP_MAX_CHANNELS, GFP_KERNEL);
if (!queue->mask)
goto no_memory_bitmap;
bitmap_set(queue->mask, 0, PTP_MAX_CHANNELS);
spin_lock_init(&queue->lock);
mutex_init(&ptp->pincfg_mux); mutex_init(&ptp->pincfg_mux);
mutex_init(&ptp->n_vclocks_mux); mutex_init(&ptp->n_vclocks_mux);
init_waitqueue_head(&ptp->tsev_wq); init_waitqueue_head(&ptp->tsev_wq);
...@@ -320,6 +342,10 @@ struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info, ...@@ -320,6 +342,10 @@ struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info,
return ERR_PTR(err); return ERR_PTR(err);
} }
/* Debugfs initialization */
sprintf(debugfsname, "ptp%d", ptp->index);
ptp->debugfs_root = debugfs_create_dir(debugfsname, NULL);
return ptp; return ptp;
no_pps: no_pps:
...@@ -330,9 +356,13 @@ struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info, ...@@ -330,9 +356,13 @@ struct ptp_clock *ptp_clock_register(struct ptp_clock_info *info,
if (ptp->kworker) if (ptp->kworker)
kthread_destroy_worker(ptp->kworker); kthread_destroy_worker(ptp->kworker);
kworker_err: kworker_err:
mutex_destroy(&ptp->tsevq_mux);
mutex_destroy(&ptp->pincfg_mux); mutex_destroy(&ptp->pincfg_mux);
mutex_destroy(&ptp->n_vclocks_mux); mutex_destroy(&ptp->n_vclocks_mux);
bitmap_free(queue->mask);
no_memory_bitmap:
list_del(&queue->qlist);
kfree(queue);
no_memory_queue:
ida_free(&ptp_clocks_map, index); ida_free(&ptp_clocks_map, index);
no_slot: no_slot:
kfree(ptp); kfree(ptp);
...@@ -375,6 +405,7 @@ EXPORT_SYMBOL(ptp_clock_unregister); ...@@ -375,6 +405,7 @@ EXPORT_SYMBOL(ptp_clock_unregister);
void ptp_clock_event(struct ptp_clock *ptp, struct ptp_clock_event *event) void ptp_clock_event(struct ptp_clock *ptp, struct ptp_clock_event *event)
{ {
struct timestamp_event_queue *tsevq;
struct pps_event_time evt; struct pps_event_time evt;
switch (event->type) { switch (event->type) {
...@@ -383,7 +414,11 @@ void ptp_clock_event(struct ptp_clock *ptp, struct ptp_clock_event *event) ...@@ -383,7 +414,11 @@ void ptp_clock_event(struct ptp_clock *ptp, struct ptp_clock_event *event)
break; break;
case PTP_CLOCK_EXTTS: case PTP_CLOCK_EXTTS:
enqueue_external_timestamp(&ptp->tsevq, event); /* Enqueue timestamp on selected queues */
list_for_each_entry(tsevq, &ptp->tsevqs, qlist) {
if (test_bit((unsigned int)event->index, tsevq->mask))
enqueue_external_timestamp(tsevq, event);
}
wake_up_interruptible(&ptp->tsev_wq); wake_up_interruptible(&ptp->tsev_wq);
break; break;
......
...@@ -15,16 +15,24 @@ ...@@ -15,16 +15,24 @@
#include <linux/ptp_clock.h> #include <linux/ptp_clock.h>
#include <linux/ptp_clock_kernel.h> #include <linux/ptp_clock_kernel.h>
#include <linux/time.h> #include <linux/time.h>
#include <linux/list.h>
#include <linux/bitmap.h>
#include <linux/debugfs.h>
#define PTP_MAX_TIMESTAMPS 128 #define PTP_MAX_TIMESTAMPS 128
#define PTP_BUF_TIMESTAMPS 30 #define PTP_BUF_TIMESTAMPS 30
#define PTP_DEFAULT_MAX_VCLOCKS 20 #define PTP_DEFAULT_MAX_VCLOCKS 20
#define PTP_MAX_CHANNELS 2048
struct timestamp_event_queue { struct timestamp_event_queue {
struct ptp_extts_event buf[PTP_MAX_TIMESTAMPS]; struct ptp_extts_event buf[PTP_MAX_TIMESTAMPS];
int head; int head;
int tail; int tail;
spinlock_t lock; spinlock_t lock;
struct list_head qlist;
unsigned long *mask;
struct dentry *debugfs_instance;
struct debugfs_u32_array dfs_bitmap;
}; };
struct ptp_clock { struct ptp_clock {
...@@ -35,8 +43,7 @@ struct ptp_clock { ...@@ -35,8 +43,7 @@ struct ptp_clock {
int index; /* index into clocks.map */ int index; /* index into clocks.map */
struct pps_device *pps_source; struct pps_device *pps_source;
long dialed_frequency; /* remembers the frequency adjustment */ long dialed_frequency; /* remembers the frequency adjustment */
struct timestamp_event_queue tsevq; /* simple fifo for time stamps */ struct list_head tsevqs; /* timestamp fifo list */
struct mutex tsevq_mux; /* one process at a time reading the fifo */
struct mutex pincfg_mux; /* protect concurrent info->pin_config access */ struct mutex pincfg_mux; /* protect concurrent info->pin_config access */
wait_queue_head_t tsev_wq; wait_queue_head_t tsev_wq;
int defunct; /* tells readers to go away when clock is being removed */ int defunct; /* tells readers to go away when clock is being removed */
...@@ -53,6 +60,7 @@ struct ptp_clock { ...@@ -53,6 +60,7 @@ struct ptp_clock {
struct mutex n_vclocks_mux; /* protect concurrent n_vclocks access */ struct mutex n_vclocks_mux; /* protect concurrent n_vclocks access */
bool is_virtual_clock; bool is_virtual_clock;
bool has_cycles; bool has_cycles;
struct dentry *debugfs_root;
}; };
#define info_to_vclock(d) container_of((d), struct ptp_vclock, info) #define info_to_vclock(d) container_of((d), struct ptp_vclock, info)
...@@ -117,16 +125,18 @@ extern struct class *ptp_class; ...@@ -117,16 +125,18 @@ extern struct class *ptp_class;
int ptp_set_pinfunc(struct ptp_clock *ptp, unsigned int pin, int ptp_set_pinfunc(struct ptp_clock *ptp, unsigned int pin,
enum ptp_pin_function func, unsigned int chan); enum ptp_pin_function func, unsigned int chan);
long ptp_ioctl(struct posix_clock *pc, long ptp_ioctl(struct posix_clock_context *pccontext, unsigned int cmd,
unsigned int cmd, unsigned long arg); unsigned long arg);
int ptp_open(struct posix_clock *pc, fmode_t fmode); int ptp_open(struct posix_clock_context *pccontext, fmode_t fmode);
ssize_t ptp_read(struct posix_clock *pc, int ptp_release(struct posix_clock_context *pccontext);
uint flags, char __user *buf, size_t cnt);
__poll_t ptp_poll(struct posix_clock *pc, ssize_t ptp_read(struct posix_clock_context *pccontext, uint flags, char __user *buf,
struct file *fp, poll_table *wait); size_t cnt);
__poll_t ptp_poll(struct posix_clock_context *pccontext, struct file *fp,
poll_table *wait);
/* /*
* see ptp_sysfs.c * see ptp_sysfs.c
......
...@@ -75,17 +75,21 @@ static ssize_t extts_fifo_show(struct device *dev, ...@@ -75,17 +75,21 @@ static ssize_t extts_fifo_show(struct device *dev,
struct device_attribute *attr, char *page) struct device_attribute *attr, char *page)
{ {
struct ptp_clock *ptp = dev_get_drvdata(dev); struct ptp_clock *ptp = dev_get_drvdata(dev);
struct timestamp_event_queue *queue = &ptp->tsevq; struct timestamp_event_queue *queue;
struct ptp_extts_event event; struct ptp_extts_event event;
unsigned long flags; unsigned long flags;
size_t qcnt; size_t qcnt;
int cnt = 0; int cnt = 0;
memset(&event, 0, sizeof(event)); cnt = list_count_nodes(&ptp->tsevqs);
if (cnt <= 0)
goto out;
if (mutex_lock_interruptible(&ptp->tsevq_mux)) /* The sysfs fifo will always draw from the fist queue */
return -ERESTARTSYS; queue = list_first_entry(&ptp->tsevqs, struct timestamp_event_queue,
qlist);
memset(&event, 0, sizeof(event));
spin_lock_irqsave(&queue->lock, flags); spin_lock_irqsave(&queue->lock, flags);
qcnt = queue_cnt(queue); qcnt = queue_cnt(queue);
if (qcnt) { if (qcnt) {
...@@ -100,7 +104,6 @@ static ssize_t extts_fifo_show(struct device *dev, ...@@ -100,7 +104,6 @@ static ssize_t extts_fifo_show(struct device *dev,
cnt = snprintf(page, PAGE_SIZE, "%u %lld %u\n", cnt = snprintf(page, PAGE_SIZE, "%u %lld %u\n",
event.index, event.t.sec, event.t.nsec); event.index, event.t.sec, event.t.nsec);
out: out:
mutex_unlock(&ptp->tsevq_mux);
return cnt; return cnt;
} }
static DEVICE_ATTR(fifo, 0444, extts_fifo_show, NULL); static DEVICE_ATTR(fifo, 0444, extts_fifo_show, NULL);
......
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
#include <linux/rwsem.h> #include <linux/rwsem.h>
struct posix_clock; struct posix_clock;
struct posix_clock_context;
/** /**
* struct posix_clock_operations - functional interface to the clock * struct posix_clock_operations - functional interface to the clock
...@@ -50,18 +51,18 @@ struct posix_clock_operations { ...@@ -50,18 +51,18 @@ struct posix_clock_operations {
/* /*
* Optional character device methods: * Optional character device methods:
*/ */
long (*ioctl) (struct posix_clock *pc, long (*ioctl)(struct posix_clock_context *pccontext, unsigned int cmd,
unsigned int cmd, unsigned long arg); unsigned long arg);
int (*open) (struct posix_clock *pc, fmode_t f_mode); int (*open)(struct posix_clock_context *pccontext, fmode_t f_mode);
__poll_t (*poll) (struct posix_clock *pc, __poll_t (*poll)(struct posix_clock_context *pccontext, struct file *file,
struct file *file, poll_table *wait); poll_table *wait);
int (*release) (struct posix_clock *pc); int (*release)(struct posix_clock_context *pccontext);
ssize_t (*read) (struct posix_clock *pc, ssize_t (*read)(struct posix_clock_context *pccontext, uint flags,
uint flags, char __user *buf, size_t cnt); char __user *buf, size_t cnt);
}; };
/** /**
...@@ -90,6 +91,24 @@ struct posix_clock { ...@@ -90,6 +91,24 @@ struct posix_clock {
bool zombie; bool zombie;
}; };
/**
* struct posix_clock_context - represents clock file operations context
*
* @clk: Pointer to the clock
* @private_clkdata: Pointer to user data
*
* Drivers should use struct posix_clock_context during specific character
* device file operation methods to access the posix clock.
*
* Drivers can store a private data structure during the open operation
* if they have specific information that is required in other file
* operations.
*/
struct posix_clock_context {
struct posix_clock *clk;
void *private_clkdata;
};
/** /**
* posix_clock_register() - register a new clock * posix_clock_register() - register a new clock
* @clk: Pointer to the clock. Caller must provide 'ops' field * @clk: Pointer to the clock. Caller must provide 'ops' field
......
...@@ -224,6 +224,8 @@ struct ptp_pin_desc { ...@@ -224,6 +224,8 @@ struct ptp_pin_desc {
_IOWR(PTP_CLK_MAGIC, 17, struct ptp_sys_offset_precise) _IOWR(PTP_CLK_MAGIC, 17, struct ptp_sys_offset_precise)
#define PTP_SYS_OFFSET_EXTENDED2 \ #define PTP_SYS_OFFSET_EXTENDED2 \
_IOWR(PTP_CLK_MAGIC, 18, struct ptp_sys_offset_extended) _IOWR(PTP_CLK_MAGIC, 18, struct ptp_sys_offset_extended)
#define PTP_MASK_CLEAR_ALL _IO(PTP_CLK_MAGIC, 19)
#define PTP_MASK_EN_SINGLE _IOW(PTP_CLK_MAGIC, 20, unsigned int)
struct ptp_extts_event { struct ptp_extts_event {
struct ptp_clock_time t; /* Time event occured. */ struct ptp_clock_time t; /* Time event occured. */
......
...@@ -19,7 +19,8 @@ ...@@ -19,7 +19,8 @@
*/ */
static struct posix_clock *get_posix_clock(struct file *fp) static struct posix_clock *get_posix_clock(struct file *fp)
{ {
struct posix_clock *clk = fp->private_data; struct posix_clock_context *pccontext = fp->private_data;
struct posix_clock *clk = pccontext->clk;
down_read(&clk->rwsem); down_read(&clk->rwsem);
...@@ -39,6 +40,7 @@ static void put_posix_clock(struct posix_clock *clk) ...@@ -39,6 +40,7 @@ static void put_posix_clock(struct posix_clock *clk)
static ssize_t posix_clock_read(struct file *fp, char __user *buf, static ssize_t posix_clock_read(struct file *fp, char __user *buf,
size_t count, loff_t *ppos) size_t count, loff_t *ppos)
{ {
struct posix_clock_context *pccontext = fp->private_data;
struct posix_clock *clk = get_posix_clock(fp); struct posix_clock *clk = get_posix_clock(fp);
int err = -EINVAL; int err = -EINVAL;
...@@ -46,7 +48,7 @@ static ssize_t posix_clock_read(struct file *fp, char __user *buf, ...@@ -46,7 +48,7 @@ static ssize_t posix_clock_read(struct file *fp, char __user *buf,
return -ENODEV; return -ENODEV;
if (clk->ops.read) if (clk->ops.read)
err = clk->ops.read(clk, fp->f_flags, buf, count); err = clk->ops.read(pccontext, fp->f_flags, buf, count);
put_posix_clock(clk); put_posix_clock(clk);
...@@ -55,6 +57,7 @@ static ssize_t posix_clock_read(struct file *fp, char __user *buf, ...@@ -55,6 +57,7 @@ static ssize_t posix_clock_read(struct file *fp, char __user *buf,
static __poll_t posix_clock_poll(struct file *fp, poll_table *wait) static __poll_t posix_clock_poll(struct file *fp, poll_table *wait)
{ {
struct posix_clock_context *pccontext = fp->private_data;
struct posix_clock *clk = get_posix_clock(fp); struct posix_clock *clk = get_posix_clock(fp);
__poll_t result = 0; __poll_t result = 0;
...@@ -62,7 +65,7 @@ static __poll_t posix_clock_poll(struct file *fp, poll_table *wait) ...@@ -62,7 +65,7 @@ static __poll_t posix_clock_poll(struct file *fp, poll_table *wait)
return EPOLLERR; return EPOLLERR;
if (clk->ops.poll) if (clk->ops.poll)
result = clk->ops.poll(clk, fp, wait); result = clk->ops.poll(pccontext, fp, wait);
put_posix_clock(clk); put_posix_clock(clk);
...@@ -72,6 +75,7 @@ static __poll_t posix_clock_poll(struct file *fp, poll_table *wait) ...@@ -72,6 +75,7 @@ static __poll_t posix_clock_poll(struct file *fp, poll_table *wait)
static long posix_clock_ioctl(struct file *fp, static long posix_clock_ioctl(struct file *fp,
unsigned int cmd, unsigned long arg) unsigned int cmd, unsigned long arg)
{ {
struct posix_clock_context *pccontext = fp->private_data;
struct posix_clock *clk = get_posix_clock(fp); struct posix_clock *clk = get_posix_clock(fp);
int err = -ENOTTY; int err = -ENOTTY;
...@@ -79,7 +83,7 @@ static long posix_clock_ioctl(struct file *fp, ...@@ -79,7 +83,7 @@ static long posix_clock_ioctl(struct file *fp,
return -ENODEV; return -ENODEV;
if (clk->ops.ioctl) if (clk->ops.ioctl)
err = clk->ops.ioctl(clk, cmd, arg); err = clk->ops.ioctl(pccontext, cmd, arg);
put_posix_clock(clk); put_posix_clock(clk);
...@@ -90,6 +94,7 @@ static long posix_clock_ioctl(struct file *fp, ...@@ -90,6 +94,7 @@ static long posix_clock_ioctl(struct file *fp,
static long posix_clock_compat_ioctl(struct file *fp, static long posix_clock_compat_ioctl(struct file *fp,
unsigned int cmd, unsigned long arg) unsigned int cmd, unsigned long arg)
{ {
struct posix_clock_context *pccontext = fp->private_data;
struct posix_clock *clk = get_posix_clock(fp); struct posix_clock *clk = get_posix_clock(fp);
int err = -ENOTTY; int err = -ENOTTY;
...@@ -97,7 +102,7 @@ static long posix_clock_compat_ioctl(struct file *fp, ...@@ -97,7 +102,7 @@ static long posix_clock_compat_ioctl(struct file *fp,
return -ENODEV; return -ENODEV;
if (clk->ops.ioctl) if (clk->ops.ioctl)
err = clk->ops.ioctl(clk, cmd, arg); err = clk->ops.ioctl(pccontext, cmd, arg);
put_posix_clock(clk); put_posix_clock(clk);
...@@ -110,6 +115,7 @@ static int posix_clock_open(struct inode *inode, struct file *fp) ...@@ -110,6 +115,7 @@ static int posix_clock_open(struct inode *inode, struct file *fp)
int err; int err;
struct posix_clock *clk = struct posix_clock *clk =
container_of(inode->i_cdev, struct posix_clock, cdev); container_of(inode->i_cdev, struct posix_clock, cdev);
struct posix_clock_context *pccontext;
down_read(&clk->rwsem); down_read(&clk->rwsem);
...@@ -117,14 +123,20 @@ static int posix_clock_open(struct inode *inode, struct file *fp) ...@@ -117,14 +123,20 @@ static int posix_clock_open(struct inode *inode, struct file *fp)
err = -ENODEV; err = -ENODEV;
goto out; goto out;
} }
pccontext = kzalloc(sizeof(*pccontext), GFP_KERNEL);
if (!pccontext) {
err = -ENOMEM;
goto out;
}
pccontext->clk = clk;
fp->private_data = pccontext;
if (clk->ops.open) if (clk->ops.open)
err = clk->ops.open(clk, fp->f_mode); err = clk->ops.open(pccontext, fp->f_mode);
else else
err = 0; err = 0;
if (!err) { if (!err) {
get_device(clk->dev); get_device(clk->dev);
fp->private_data = clk;
} }
out: out:
up_read(&clk->rwsem); up_read(&clk->rwsem);
...@@ -133,14 +145,20 @@ static int posix_clock_open(struct inode *inode, struct file *fp) ...@@ -133,14 +145,20 @@ static int posix_clock_open(struct inode *inode, struct file *fp)
static int posix_clock_release(struct inode *inode, struct file *fp) static int posix_clock_release(struct inode *inode, struct file *fp)
{ {
struct posix_clock *clk = fp->private_data; struct posix_clock_context *pccontext = fp->private_data;
struct posix_clock *clk;
int err = 0; int err = 0;
if (!pccontext)
return -ENODEV;
clk = pccontext->clk;
if (clk->ops.release) if (clk->ops.release)
err = clk->ops.release(clk); err = clk->ops.release(pccontext);
put_device(clk->dev); put_device(clk->dev);
kfree(pccontext);
fp->private_data = NULL; fp->private_data = NULL;
return err; return err;
......
#!/bin/bash
# SPDX-License-Identifier: GPL-2.0
# Simple helper script to transform ptp debugfs timestamp event queue filtering
# masks from decimal values to hexadecimal values
# Only takes the debugfs mask file path as an argument
DEBUGFS_MASKFILE="${1}"
#shellcheck disable=SC2013,SC2086
for int in $(cat "$DEBUGFS_MASKFILE") ; do
printf '0x%08X ' "$int"
done
echo
...@@ -121,6 +121,7 @@ static void usage(char *progname) ...@@ -121,6 +121,7 @@ static void usage(char *progname)
" -d name device to open\n" " -d name device to open\n"
" -e val read 'val' external time stamp events\n" " -e val read 'val' external time stamp events\n"
" -f val adjust the ptp clock frequency by 'val' ppb\n" " -f val adjust the ptp clock frequency by 'val' ppb\n"
" -F chan Enable single channel mask and keep device open for debugfs verification.\n"
" -g get the ptp clock time\n" " -g get the ptp clock time\n"
" -h prints this message\n" " -h prints this message\n"
" -i val index for event/trigger\n" " -i val index for event/trigger\n"
...@@ -187,6 +188,7 @@ int main(int argc, char *argv[]) ...@@ -187,6 +188,7 @@ int main(int argc, char *argv[])
int pps = -1; int pps = -1;
int seconds = 0; int seconds = 0;
int settime = 0; int settime = 0;
int channel = -1;
int64_t t1, t2, tp; int64_t t1, t2, tp;
int64_t interval, offset; int64_t interval, offset;
...@@ -196,7 +198,7 @@ int main(int argc, char *argv[]) ...@@ -196,7 +198,7 @@ int main(int argc, char *argv[])
progname = strrchr(argv[0], '/'); progname = strrchr(argv[0], '/');
progname = progname ? 1+progname : argv[0]; progname = progname ? 1+progname : argv[0];
while (EOF != (c = getopt(argc, argv, "cd:e:f:ghH:i:k:lL:n:o:p:P:sSt:T:w:x:Xz"))) { while (EOF != (c = getopt(argc, argv, "cd:e:f:F:ghH:i:k:lL:n:o:p:P:sSt:T:w:x:Xz"))) {
switch (c) { switch (c) {
case 'c': case 'c':
capabilities = 1; capabilities = 1;
...@@ -210,6 +212,9 @@ int main(int argc, char *argv[]) ...@@ -210,6 +212,9 @@ int main(int argc, char *argv[])
case 'f': case 'f':
adjfreq = atoi(optarg); adjfreq = atoi(optarg);
break; break;
case 'F':
channel = atoi(optarg);
break;
case 'g': case 'g':
gettime = 1; gettime = 1;
break; break;
...@@ -604,6 +609,18 @@ int main(int argc, char *argv[]) ...@@ -604,6 +609,18 @@ int main(int argc, char *argv[])
free(xts); free(xts);
} }
if (channel >= 0) {
if (ioctl(fd, PTP_MASK_CLEAR_ALL)) {
perror("PTP_MASK_CLEAR_ALL");
} else if (ioctl(fd, PTP_MASK_EN_SINGLE, (unsigned int *)&channel)) {
perror("PTP_MASK_EN_SINGLE");
} else {
printf("Channel %d exclusively enabled. Check on debugfs.\n", channel);
printf("Press any key to continue\n.");
getchar();
}
}
close(fd); close(fd);
return 0; return 0;
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment