Commit 8a228ecf authored by Bjorn Andersson's avatar Bjorn Andersson

rpmsg: Indirection table for rpmsg_endpoint operations

Add indirection table for rpmsg_endpoint related operations and move
virtio implementation behind this, this finishes of the decoupling of
the virtio implementation from the public API.
Signed-off-by: default avatarBjorn Andersson <bjorn.andersson@linaro.org>
parent 026dad47
...@@ -107,6 +107,18 @@ struct virtproc_info { ...@@ -107,6 +107,18 @@ struct virtproc_info {
/* Address 53 is reserved for advertising remote services */ /* Address 53 is reserved for advertising remote services */
#define RPMSG_NS_ADDR (53) #define RPMSG_NS_ADDR (53)
static void virtio_rpmsg_destroy_ept(struct rpmsg_endpoint *ept);
static int virtio_rpmsg_send(struct rpmsg_endpoint *ept, void *data, int len);
static int virtio_rpmsg_sendto(struct rpmsg_endpoint *ept, void *data, int len,
u32 dst);
static int virtio_rpmsg_send_offchannel(struct rpmsg_endpoint *ept, u32 src,
u32 dst, void *data, int len);
static int virtio_rpmsg_trysend(struct rpmsg_endpoint *ept, void *data, int len);
static int virtio_rpmsg_trysendto(struct rpmsg_endpoint *ept, void *data,
int len, u32 dst);
static int virtio_rpmsg_trysend_offchannel(struct rpmsg_endpoint *ept, u32 src,
u32 dst, void *data, int len);
/* sysfs show configuration fields */ /* sysfs show configuration fields */
#define rpmsg_show_attr(field, path, format_string) \ #define rpmsg_show_attr(field, path, format_string) \
static ssize_t \ static ssize_t \
...@@ -172,6 +184,16 @@ static int rpmsg_uevent(struct device *dev, struct kobj_uevent_env *env) ...@@ -172,6 +184,16 @@ static int rpmsg_uevent(struct device *dev, struct kobj_uevent_env *env)
rpdev->id.name); rpdev->id.name);
} }
static const struct rpmsg_endpoint_ops virtio_endpoint_ops = {
.destroy_ept = virtio_rpmsg_destroy_ept,
.send = virtio_rpmsg_send,
.sendto = virtio_rpmsg_sendto,
.send_offchannel = virtio_rpmsg_send_offchannel,
.trysend = virtio_rpmsg_trysend,
.trysendto = virtio_rpmsg_trysendto,
.trysend_offchannel = virtio_rpmsg_trysend_offchannel,
};
/** /**
* __ept_release() - deallocate an rpmsg endpoint * __ept_release() - deallocate an rpmsg endpoint
* @kref: the ept's reference count * @kref: the ept's reference count
...@@ -212,6 +234,7 @@ static struct rpmsg_endpoint *__rpmsg_create_ept(struct virtproc_info *vrp, ...@@ -212,6 +234,7 @@ static struct rpmsg_endpoint *__rpmsg_create_ept(struct virtproc_info *vrp,
ept->rpdev = rpdev; ept->rpdev = rpdev;
ept->cb = cb; ept->cb = cb;
ept->priv = priv; ept->priv = priv;
ept->ops = &virtio_endpoint_ops;
/* do we need to allocate a local address ? */ /* do we need to allocate a local address ? */
if (addr == RPMSG_ADDR_ANY) { if (addr == RPMSG_ADDR_ANY) {
...@@ -285,10 +308,15 @@ __rpmsg_destroy_ept(struct virtproc_info *vrp, struct rpmsg_endpoint *ept) ...@@ -285,10 +308,15 @@ __rpmsg_destroy_ept(struct virtproc_info *vrp, struct rpmsg_endpoint *ept)
*/ */
void rpmsg_destroy_ept(struct rpmsg_endpoint *ept) void rpmsg_destroy_ept(struct rpmsg_endpoint *ept)
{ {
__rpmsg_destroy_ept(ept->rpdev->vrp, ept); ept->ops->destroy_ept(ept);
} }
EXPORT_SYMBOL(rpmsg_destroy_ept); EXPORT_SYMBOL(rpmsg_destroy_ept);
static void virtio_rpmsg_destroy_ept(struct rpmsg_endpoint *ept)
{
__rpmsg_destroy_ept(ept->rpdev->vrp, ept);
}
/* /*
* when an rpmsg driver is probed with a channel, we seamlessly create * when an rpmsg driver is probed with a channel, we seamlessly create
* it an endpoint, binding its rx callback to a unique local rpmsg * it an endpoint, binding its rx callback to a unique local rpmsg
...@@ -657,8 +685,9 @@ static void rpmsg_downref_sleepers(struct virtproc_info *vrp) ...@@ -657,8 +685,9 @@ static void rpmsg_downref_sleepers(struct virtproc_info *vrp)
* *
* Returns 0 on success and an appropriate error value on failure. * Returns 0 on success and an appropriate error value on failure.
*/ */
int rpmsg_send_offchannel_raw(struct rpmsg_device *rpdev, u32 src, u32 dst, static int rpmsg_send_offchannel_raw(struct rpmsg_device *rpdev,
void *data, int len, bool wait) u32 src, u32 dst,
void *data, int len, bool wait)
{ {
struct virtproc_info *vrp = rpdev->vrp; struct virtproc_info *vrp = rpdev->vrp;
struct device *dev = &rpdev->dev; struct device *dev = &rpdev->dev;
...@@ -754,6 +783,56 @@ int rpmsg_send_offchannel_raw(struct rpmsg_device *rpdev, u32 src, u32 dst, ...@@ -754,6 +783,56 @@ int rpmsg_send_offchannel_raw(struct rpmsg_device *rpdev, u32 src, u32 dst,
} }
EXPORT_SYMBOL(rpmsg_send_offchannel_raw); EXPORT_SYMBOL(rpmsg_send_offchannel_raw);
static int virtio_rpmsg_send(struct rpmsg_endpoint *ept, void *data, int len)
{
struct rpmsg_device *rpdev = ept->rpdev;
u32 src = ept->addr, dst = rpdev->dst;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, true);
}
static int virtio_rpmsg_sendto(struct rpmsg_endpoint *ept, void *data, int len,
u32 dst)
{
struct rpmsg_device *rpdev = ept->rpdev;
u32 src = ept->addr;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, true);
}
static int virtio_rpmsg_send_offchannel(struct rpmsg_endpoint *ept, u32 src,
u32 dst, void *data, int len)
{
struct rpmsg_device *rpdev = ept->rpdev;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, true);
}
static int virtio_rpmsg_trysend(struct rpmsg_endpoint *ept, void *data, int len)
{
struct rpmsg_device *rpdev = ept->rpdev;
u32 src = ept->addr, dst = rpdev->dst;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, false);
}
static int virtio_rpmsg_trysendto(struct rpmsg_endpoint *ept, void *data,
int len, u32 dst)
{
struct rpmsg_device *rpdev = ept->rpdev;
u32 src = ept->addr;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, false);
}
static int virtio_rpmsg_trysend_offchannel(struct rpmsg_endpoint *ept, u32 src,
u32 dst, void *data, int len)
{
struct rpmsg_device *rpdev = ept->rpdev;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, false);
}
static int rpmsg_recv_single(struct virtproc_info *vrp, struct device *dev, static int rpmsg_recv_single(struct virtproc_info *vrp, struct device *dev,
struct rpmsg_hdr *msg, unsigned int len) struct rpmsg_hdr *msg, unsigned int len)
{ {
......
...@@ -96,8 +96,10 @@ enum rpmsg_ns_flags { ...@@ -96,8 +96,10 @@ enum rpmsg_ns_flags {
#define RPMSG_ADDR_ANY 0xFFFFFFFF #define RPMSG_ADDR_ANY 0xFFFFFFFF
struct virtproc_info; struct virtproc_info;
struct rpmsg_device;
struct rpmsg_endpoint; struct rpmsg_endpoint;
struct rpmsg_device_ops; struct rpmsg_device_ops;
struct rpmsg_endpoint_ops;
/** /**
* struct rpmsg_channel_info - channel info representation * struct rpmsg_channel_info - channel info representation
...@@ -184,6 +186,36 @@ struct rpmsg_endpoint { ...@@ -184,6 +186,36 @@ struct rpmsg_endpoint {
struct mutex cb_lock; struct mutex cb_lock;
u32 addr; u32 addr;
void *priv; void *priv;
const struct rpmsg_endpoint_ops *ops;
};
/**
* struct rpmsg_endpoint_ops - indirection table for rpmsg_endpoint operations
* @destroy_ept: destroy the given endpoint, required
* @send: see @rpmsg_send(), required
* @sendto: see @rpmsg_sendto(), optional
* @send_offchannel: see @rpmsg_send_offchannel(), optional
* @trysend: see @rpmsg_trysend(), required
* @trysendto: see @rpmsg_trysendto(), optional
* @trysend_offchannel: see @rpmsg_trysend_offchannel(), optional
*
* Indirection table for the operations that a rpmsg backend should implement.
* In addition to @destroy_ept, the backend must at least implement @send and
* @trysend, while the variants sending data off-channel are optional.
*/
struct rpmsg_endpoint_ops {
void (*destroy_ept)(struct rpmsg_endpoint *ept);
int (*send)(struct rpmsg_endpoint *ept, void *data, int len);
int (*sendto)(struct rpmsg_endpoint *ept, void *data, int len, u32 dst);
int (*send_offchannel)(struct rpmsg_endpoint *ept, u32 src, u32 dst,
void *data, int len);
int (*trysend)(struct rpmsg_endpoint *ept, void *data, int len);
int (*trysendto)(struct rpmsg_endpoint *ept, void *data, int len, u32 dst);
int (*trysend_offchannel)(struct rpmsg_endpoint *ept, u32 src, u32 dst,
void *data, int len);
}; };
/** /**
...@@ -210,8 +242,6 @@ void rpmsg_destroy_ept(struct rpmsg_endpoint *); ...@@ -210,8 +242,6 @@ void rpmsg_destroy_ept(struct rpmsg_endpoint *);
struct rpmsg_endpoint *rpmsg_create_ept(struct rpmsg_device *, struct rpmsg_endpoint *rpmsg_create_ept(struct rpmsg_device *,
rpmsg_rx_cb_t cb, void *priv, rpmsg_rx_cb_t cb, void *priv,
struct rpmsg_channel_info chinfo); struct rpmsg_channel_info chinfo);
int
rpmsg_send_offchannel_raw(struct rpmsg_device *, u32, u32, void *, int, bool);
/* use a macro to avoid include chaining to get THIS_MODULE */ /* use a macro to avoid include chaining to get THIS_MODULE */
#define register_rpmsg_driver(drv) \ #define register_rpmsg_driver(drv) \
...@@ -249,10 +279,7 @@ rpmsg_send_offchannel_raw(struct rpmsg_device *, u32, u32, void *, int, bool); ...@@ -249,10 +279,7 @@ rpmsg_send_offchannel_raw(struct rpmsg_device *, u32, u32, void *, int, bool);
*/ */
static inline int rpmsg_send(struct rpmsg_endpoint *ept, void *data, int len) static inline int rpmsg_send(struct rpmsg_endpoint *ept, void *data, int len)
{ {
struct rpmsg_device *rpdev = ept->rpdev; return ept->ops->send(ept, data, len);
u32 src = ept->addr, dst = rpdev->dst;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, true);
} }
/** /**
...@@ -276,10 +303,7 @@ static inline int rpmsg_send(struct rpmsg_endpoint *ept, void *data, int len) ...@@ -276,10 +303,7 @@ static inline int rpmsg_send(struct rpmsg_endpoint *ept, void *data, int len)
static inline static inline
int rpmsg_sendto(struct rpmsg_endpoint *ept, void *data, int len, u32 dst) int rpmsg_sendto(struct rpmsg_endpoint *ept, void *data, int len, u32 dst)
{ {
struct rpmsg_device *rpdev = ept->rpdev; return ept->ops->sendto(ept, data, len, dst);
u32 src = ept->addr;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, true);
} }
/** /**
...@@ -306,9 +330,7 @@ static inline ...@@ -306,9 +330,7 @@ static inline
int rpmsg_send_offchannel(struct rpmsg_endpoint *ept, u32 src, u32 dst, int rpmsg_send_offchannel(struct rpmsg_endpoint *ept, u32 src, u32 dst,
void *data, int len) void *data, int len)
{ {
struct rpmsg_device *rpdev = ept->rpdev; return ept->ops->send_offchannel(ept, src, dst, data, len);
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, true);
} }
/** /**
...@@ -331,10 +353,7 @@ int rpmsg_send_offchannel(struct rpmsg_endpoint *ept, u32 src, u32 dst, ...@@ -331,10 +353,7 @@ int rpmsg_send_offchannel(struct rpmsg_endpoint *ept, u32 src, u32 dst,
static inline static inline
int rpmsg_trysend(struct rpmsg_endpoint *ept, void *data, int len) int rpmsg_trysend(struct rpmsg_endpoint *ept, void *data, int len)
{ {
struct rpmsg_device *rpdev = ept->rpdev; return ept->ops->trysend(ept, data, len);
u32 src = ept->addr, dst = rpdev->dst;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, false);
} }
/** /**
...@@ -357,10 +376,7 @@ int rpmsg_trysend(struct rpmsg_endpoint *ept, void *data, int len) ...@@ -357,10 +376,7 @@ int rpmsg_trysend(struct rpmsg_endpoint *ept, void *data, int len)
static inline static inline
int rpmsg_trysendto(struct rpmsg_endpoint *ept, void *data, int len, u32 dst) int rpmsg_trysendto(struct rpmsg_endpoint *ept, void *data, int len, u32 dst)
{ {
struct rpmsg_device *rpdev = ept->rpdev; return ept->ops->trysendto(ept, data, len, dst);
u32 src = ept->addr;
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, false);
} }
/** /**
...@@ -386,9 +402,7 @@ static inline ...@@ -386,9 +402,7 @@ static inline
int rpmsg_trysend_offchannel(struct rpmsg_endpoint *ept, u32 src, u32 dst, int rpmsg_trysend_offchannel(struct rpmsg_endpoint *ept, u32 src, u32 dst,
void *data, int len) void *data, int len)
{ {
struct rpmsg_device *rpdev = ept->rpdev; return ept->ops->trysend_offchannel(ept, src, dst, data, len);
return rpmsg_send_offchannel_raw(rpdev, src, dst, data, len, false);
} }
#endif /* _LINUX_RPMSG_H */ #endif /* _LINUX_RPMSG_H */
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