#include <linux/device.h>
#include <linux/err.h>
#include <linux/ptp_clock_kernel.h>
#include <uapi/linux/virtio_rtc.h>
#include "virtio_rtc_internal.h"
struct viortc_ptp_clock {
struct ptp_clock *ptp_clock;
struct viortc_dev *viortc;
struct ptp_clock_info ptp_info;
u16 vio_clk_id;
bool have_cross;
};
struct viortc_ptp_cross_ctx {
ktime_t device_time;
struct system_counterval_t system_counterval;
};
int __weak viortc_hw_xtstamp_params(u8 *hw_counter, enum clocksource_ids *cs_id)
{
return -EOPNOTSUPP;
}
static int viortc_ptp_get_time_fn(ktime_t *device_time,
struct system_counterval_t *system_counterval,
void *ctx)
{
struct viortc_ptp_cross_ctx *vio_ctx = ctx;
*device_time = vio_ctx->device_time;
*system_counterval = vio_ctx->system_counterval;
return 0;
}
static int viortc_ptp_do_xtstamp(struct viortc_ptp_clock *vio_ptp,
u8 hw_counter, enum clocksource_ids cs_id,
struct viortc_ptp_cross_ctx *ctx)
{
u64 max_ns, ns;
int ret;
ctx->system_counterval.cs_id = cs_id;
ret = viortc_read_cross(vio_ptp->viortc, vio_ptp->vio_clk_id,
hw_counter, &ns,
&ctx->system_counterval.cycles);
if (ret)
return ret;
max_ns = (u64)ktime_to_ns(KTIME_MAX);
if (ns > max_ns)
return -EINVAL;
ctx->device_time = ns_to_ktime(ns);
return 0;
}
static int viortc_ptp_getcrosststamp(struct ptp_clock_info *ptp,
struct system_device_crosststamp *xtstamp)
{
struct viortc_ptp_clock *vio_ptp =
container_of(ptp, struct viortc_ptp_clock, ptp_info);
struct system_time_snapshot history_begin;
struct viortc_ptp_cross_ctx ctx;
enum clocksource_ids cs_id;
u8 hw_counter;
int ret;
if (!vio_ptp->have_cross)
return -EOPNOTSUPP;
ret = viortc_hw_xtstamp_params(&hw_counter, &cs_id);
if (ret)
return ret;
ktime_get_snapshot(&history_begin);
if (history_begin.cs_id != cs_id)
return -EOPNOTSUPP;
ret = viortc_ptp_do_xtstamp(vio_ptp, hw_counter, cs_id, &ctx);
if (ret)
return ret;
ret = get_device_system_crosststamp(viortc_ptp_get_time_fn, &ctx,
&history_begin, xtstamp);
if (ret)
pr_debug("%s: get_device_system_crosststamp() returned %d\n",
__func__, ret);
return ret;
}
static int viortc_ptp_adjfine(struct ptp_clock_info *ptp, long scaled_ppm)
{
return -EOPNOTSUPP;
}
static int viortc_ptp_adjtime(struct ptp_clock_info *ptp, s64 delta)
{
return -EOPNOTSUPP;
}
static int viortc_ptp_settime64(struct ptp_clock_info *ptp,
const struct timespec64 *ts)
{
return -EOPNOTSUPP;
}
static int viortc_ptp_gettimex64(struct ptp_clock_info *ptp,
struct timespec64 *ts,
struct ptp_system_timestamp *sts)
{
struct viortc_ptp_clock *vio_ptp =
container_of(ptp, struct viortc_ptp_clock, ptp_info);
int ret;
u64 ns;
ptp_read_system_prets(sts);
ret = viortc_read(vio_ptp->viortc, vio_ptp->vio_clk_id, &ns);
ptp_read_system_postts(sts);
if (ret)
return ret;
if (ns > (u64)S64_MAX)
return -EINVAL;
*ts = ns_to_timespec64((s64)ns);
return 0;
}
static int viortc_ptp_enable(struct ptp_clock_info *ptp,
struct ptp_clock_request *rq, int on)
{
return -EOPNOTSUPP;
}
static const struct ptp_clock_info viortc_ptp_info_template = {
.owner = THIS_MODULE,
.adjfine = viortc_ptp_adjfine,
.adjtime = viortc_ptp_adjtime,
.gettimex64 = viortc_ptp_gettimex64,
.settime64 = viortc_ptp_settime64,
.enable = viortc_ptp_enable,
.getcrosststamp = viortc_ptp_getcrosststamp,
};
int viortc_ptp_unregister(struct viortc_ptp_clock *vio_ptp,
struct device *parent_dev)
{
int ret = ptp_clock_unregister(vio_ptp->ptp_clock);
if (!ret)
devm_kfree(parent_dev, vio_ptp);
return ret;
}
static int viortc_ptp_get_cross_cap(struct viortc_dev *viortc,
struct viortc_ptp_clock *vio_ptp)
{
enum clocksource_ids cs_id;
bool xtstamp_supported;
u8 hw_counter;
int ret;
ret = viortc_hw_xtstamp_params(&hw_counter, &cs_id);
if (ret) {
vio_ptp->have_cross = false;
return 0;
}
ret = viortc_cross_cap(viortc, vio_ptp->vio_clk_id, hw_counter,
&xtstamp_supported);
if (ret)
return ret;
vio_ptp->have_cross = xtstamp_supported;
return 0;
}
struct viortc_ptp_clock *viortc_ptp_register(struct viortc_dev *viortc,
struct device *parent_dev,
u16 vio_clk_id,
const char *ptp_clock_name)
{
struct viortc_ptp_clock *vio_ptp;
struct ptp_clock *ptp_clock;
ssize_t len;
int ret;
vio_ptp = devm_kzalloc(parent_dev, sizeof(*vio_ptp), GFP_KERNEL);
if (!vio_ptp)
return ERR_PTR(-ENOMEM);
vio_ptp->viortc = viortc;
vio_ptp->vio_clk_id = vio_clk_id;
vio_ptp->ptp_info = viortc_ptp_info_template;
len = strscpy(vio_ptp->ptp_info.name, ptp_clock_name,
sizeof(vio_ptp->ptp_info.name));
if (len < 0) {
ret = len;
goto err_free_dev;
}
ret = viortc_ptp_get_cross_cap(viortc, vio_ptp);
if (ret)
goto err_free_dev;
if (!vio_ptp->have_cross)
vio_ptp->ptp_info.getcrosststamp = NULL;
ptp_clock = ptp_clock_register(&vio_ptp->ptp_info, parent_dev);
if (IS_ERR(ptp_clock))
goto err_on_register;
vio_ptp->ptp_clock = ptp_clock;
return vio_ptp;
err_on_register:
ret = PTR_ERR(ptp_clock);
err_free_dev:
devm_kfree(parent_dev, vio_ptp);
return ERR_PTR(ret);
}