272 lines
7.4 KiB
C
272 lines
7.4 KiB
C
|
/* SPDX-License-Identifier: GPL-2.0 */
|
||
|
|
||
|
#ifndef __LINUX_HTE_H
|
||
|
#define __LINUX_HTE_H
|
||
|
|
||
|
#include <linux/errno.h>
|
||
|
|
||
|
struct hte_chip;
|
||
|
struct hte_device;
|
||
|
struct of_phandle_args;
|
||
|
|
||
|
/**
|
||
|
* enum hte_edge - HTE line edge flags.
|
||
|
*
|
||
|
* @HTE_EDGE_NO_SETUP: No edge setup. In this case consumer will setup edges,
|
||
|
* for example during request irq call.
|
||
|
* @HTE_RISING_EDGE_TS: Rising edge.
|
||
|
* @HTE_FALLING_EDGE_TS: Falling edge.
|
||
|
*
|
||
|
*/
|
||
|
enum hte_edge {
|
||
|
HTE_EDGE_NO_SETUP = 1U << 0,
|
||
|
HTE_RISING_EDGE_TS = 1U << 1,
|
||
|
HTE_FALLING_EDGE_TS = 1U << 2,
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* enum hte_return - HTE subsystem return values used during callback.
|
||
|
*
|
||
|
* @HTE_CB_HANDLED: The consumer handled the data.
|
||
|
* @HTE_RUN_SECOND_CB: The consumer needs further processing, in that case
|
||
|
* HTE subsystem calls secondary callback provided by the consumer where it
|
||
|
* is allowed to sleep.
|
||
|
*/
|
||
|
enum hte_return {
|
||
|
HTE_CB_HANDLED,
|
||
|
HTE_RUN_SECOND_CB,
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* struct hte_ts_data - HTE timestamp data.
|
||
|
*
|
||
|
* @tsc: Timestamp value.
|
||
|
* @seq: Sequence counter of the timestamps.
|
||
|
* @raw_level: Level of the line at the timestamp if provider supports it,
|
||
|
* -1 otherwise.
|
||
|
*/
|
||
|
struct hte_ts_data {
|
||
|
u64 tsc;
|
||
|
u64 seq;
|
||
|
int raw_level;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* struct hte_clk_info - Clock source info that HTE provider uses to timestamp.
|
||
|
*
|
||
|
* @hz: Supported clock rate in HZ, for example 1KHz clock = 1000.
|
||
|
* @type: Supported clock type.
|
||
|
*/
|
||
|
struct hte_clk_info {
|
||
|
u64 hz;
|
||
|
clockid_t type;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* typedef hte_ts_cb_t - HTE timestamp data processing primary callback.
|
||
|
*
|
||
|
* The callback is used to push timestamp data to the client and it is
|
||
|
* not allowed to sleep.
|
||
|
*
|
||
|
* @ts: HW timestamp data.
|
||
|
* @data: Client supplied data.
|
||
|
*/
|
||
|
typedef enum hte_return (*hte_ts_cb_t)(struct hte_ts_data *ts, void *data);
|
||
|
|
||
|
/**
|
||
|
* typedef hte_ts_sec_cb_t - HTE timestamp data processing secondary callback.
|
||
|
*
|
||
|
* This is used when the client needs further processing where it is
|
||
|
* allowed to sleep.
|
||
|
*
|
||
|
* @data: Client supplied data.
|
||
|
*
|
||
|
*/
|
||
|
typedef enum hte_return (*hte_ts_sec_cb_t)(void *data);
|
||
|
|
||
|
/**
|
||
|
* struct hte_line_attr - Line attributes.
|
||
|
*
|
||
|
* @line_id: The logical ID understood by the consumers and providers.
|
||
|
* @line_data: Line data related to line_id.
|
||
|
* @edge_flags: Edge setup flags.
|
||
|
* @name: Descriptive name of the entity that is being monitored for the
|
||
|
* hardware timestamping. If null, HTE core will construct the name.
|
||
|
*
|
||
|
*/
|
||
|
struct hte_line_attr {
|
||
|
u32 line_id;
|
||
|
void *line_data;
|
||
|
unsigned long edge_flags;
|
||
|
const char *name;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* struct hte_ts_desc - HTE timestamp descriptor.
|
||
|
*
|
||
|
* This structure is a communication token between consumers to subsystem
|
||
|
* and subsystem to providers.
|
||
|
*
|
||
|
* @attr: The line attributes.
|
||
|
* @hte_data: Subsystem's private data, set by HTE subsystem.
|
||
|
*/
|
||
|
struct hte_ts_desc {
|
||
|
struct hte_line_attr attr;
|
||
|
void *hte_data;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* struct hte_ops - HTE operations set by providers.
|
||
|
*
|
||
|
* @request: Hook for requesting a HTE timestamp. Returns 0 on success,
|
||
|
* non-zero for failures.
|
||
|
* @release: Hook for releasing a HTE timestamp. Returns 0 on success,
|
||
|
* non-zero for failures.
|
||
|
* @enable: Hook to enable the specified timestamp. Returns 0 on success,
|
||
|
* non-zero for failures.
|
||
|
* @disable: Hook to disable specified timestamp. Returns 0 on success,
|
||
|
* non-zero for failures.
|
||
|
* @get_clk_src_info: Hook to get the clock information the provider uses
|
||
|
* to timestamp. Returns 0 for success and negative error code for failure. On
|
||
|
* success HTE subsystem fills up provided struct hte_clk_info.
|
||
|
*
|
||
|
* xlated_id parameter is used to communicate between HTE subsystem and the
|
||
|
* providers and is translated by the provider.
|
||
|
*/
|
||
|
struct hte_ops {
|
||
|
int (*request)(struct hte_chip *chip, struct hte_ts_desc *desc,
|
||
|
u32 xlated_id);
|
||
|
int (*release)(struct hte_chip *chip, struct hte_ts_desc *desc,
|
||
|
u32 xlated_id);
|
||
|
int (*enable)(struct hte_chip *chip, u32 xlated_id);
|
||
|
int (*disable)(struct hte_chip *chip, u32 xlated_id);
|
||
|
int (*get_clk_src_info)(struct hte_chip *chip,
|
||
|
struct hte_clk_info *ci);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* struct hte_chip - Abstract HTE chip.
|
||
|
*
|
||
|
* @name: functional name of the HTE IP block.
|
||
|
* @dev: device providing the HTE.
|
||
|
* @ops: callbacks for this HTE.
|
||
|
* @nlines: number of lines/signals supported by this chip.
|
||
|
* @xlate_of: Callback which translates consumer supplied logical ids to
|
||
|
* physical ids, return 0 for the success and negative for the failures.
|
||
|
* It stores (between 0 to @nlines) in xlated_id parameter for the success.
|
||
|
* @xlate_plat: Same as above but for the consumers with no DT node.
|
||
|
* @match_from_linedata: Match HTE device using the line_data.
|
||
|
* @of_hte_n_cells: Number of cells used to form the HTE specifier.
|
||
|
* @gdev: HTE subsystem abstract device, internal to the HTE subsystem.
|
||
|
* @data: chip specific private data.
|
||
|
*/
|
||
|
struct hte_chip {
|
||
|
const char *name;
|
||
|
struct device *dev;
|
||
|
const struct hte_ops *ops;
|
||
|
u32 nlines;
|
||
|
int (*xlate_of)(struct hte_chip *gc,
|
||
|
const struct of_phandle_args *args,
|
||
|
struct hte_ts_desc *desc, u32 *xlated_id);
|
||
|
int (*xlate_plat)(struct hte_chip *gc, struct hte_ts_desc *desc,
|
||
|
u32 *xlated_id);
|
||
|
bool (*match_from_linedata)(const struct hte_chip *chip,
|
||
|
const struct hte_ts_desc *hdesc);
|
||
|
u8 of_hte_n_cells;
|
||
|
|
||
|
struct hte_device *gdev;
|
||
|
void *data;
|
||
|
};
|
||
|
|
||
|
#if IS_ENABLED(CONFIG_HTE)
|
||
|
/* HTE APIs for the providers */
|
||
|
int devm_hte_register_chip(struct hte_chip *chip);
|
||
|
int hte_push_ts_ns(const struct hte_chip *chip, u32 xlated_id,
|
||
|
struct hte_ts_data *data);
|
||
|
|
||
|
/* HTE APIs for the consumers */
|
||
|
int hte_init_line_attr(struct hte_ts_desc *desc, u32 line_id,
|
||
|
unsigned long edge_flags, const char *name,
|
||
|
void *data);
|
||
|
int hte_ts_get(struct device *dev, struct hte_ts_desc *desc, int index);
|
||
|
int hte_ts_put(struct hte_ts_desc *desc);
|
||
|
int hte_request_ts_ns(struct hte_ts_desc *desc, hte_ts_cb_t cb,
|
||
|
hte_ts_sec_cb_t tcb, void *data);
|
||
|
int devm_hte_request_ts_ns(struct device *dev, struct hte_ts_desc *desc,
|
||
|
hte_ts_cb_t cb, hte_ts_sec_cb_t tcb, void *data);
|
||
|
int of_hte_req_count(struct device *dev);
|
||
|
int hte_enable_ts(struct hte_ts_desc *desc);
|
||
|
int hte_disable_ts(struct hte_ts_desc *desc);
|
||
|
int hte_get_clk_src_info(const struct hte_ts_desc *desc,
|
||
|
struct hte_clk_info *ci);
|
||
|
|
||
|
#else /* !CONFIG_HTE */
|
||
|
static inline int devm_hte_register_chip(struct hte_chip *chip)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int hte_push_ts_ns(const struct hte_chip *chip,
|
||
|
u32 xlated_id,
|
||
|
const struct hte_ts_data *data)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int hte_init_line_attr(struct hte_ts_desc *desc, u32 line_id,
|
||
|
unsigned long edge_flags,
|
||
|
const char *name, void *data)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int hte_ts_get(struct device *dev, struct hte_ts_desc *desc,
|
||
|
int index)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int hte_ts_put(struct hte_ts_desc *desc)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int hte_request_ts_ns(struct hte_ts_desc *desc, hte_ts_cb_t cb,
|
||
|
hte_ts_sec_cb_t tcb, void *data)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int devm_hte_request_ts_ns(struct device *dev,
|
||
|
struct hte_ts_desc *desc,
|
||
|
hte_ts_cb_t cb,
|
||
|
hte_ts_sec_cb_t tcb,
|
||
|
void *data)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int of_hte_req_count(struct device *dev)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int hte_enable_ts(struct hte_ts_desc *desc)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int hte_disable_ts(struct hte_ts_desc *desc)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
|
||
|
static inline int hte_get_clk_src_info(const struct hte_ts_desc *desc,
|
||
|
struct hte_clk_info *ci)
|
||
|
{
|
||
|
return -EOPNOTSUPP;
|
||
|
}
|
||
|
#endif /* !CONFIG_HTE */
|
||
|
|
||
|
#endif
|