net: qrtr: Send HELLO message on endpoint register
Hello message is currently sent in response to a hello message received by the ns. When two procs operate in this slave model, then neither of them exchange the Hello message. This will halt further communication from happening. Update QRTR to send a hello message once a new endpoint is registered. Prevent duplicate Hello messages from reaching the NS because NS will echo the packet back. On two systems running QRTR, this will loopback infinitely. Change-Id: Ibc84fc46658e8c6cedb0d6e84bb1a56d739d5a30 Signed-off-by: Chris Lew <clew@codeaurora.org>
This commit is contained in:
parent
f75836aa9e
commit
3e534ba0d9
2 changed files with 75 additions and 9 deletions
|
@ -171,12 +171,11 @@ static int qcom_mhi_qrtr_probe(struct mhi_device *mhi_dev,
|
|||
INIT_LIST_HEAD(&qdev->ul_pkts);
|
||||
spin_lock_init(&qdev->ul_lock);
|
||||
|
||||
dev_set_drvdata(&mhi_dev->dev, qdev);
|
||||
rc = qrtr_endpoint_register(&qdev->ep, net_id, rt);
|
||||
if (rc)
|
||||
return rc;
|
||||
|
||||
dev_set_drvdata(&mhi_dev->dev, qdev);
|
||||
|
||||
dev_dbg(qdev->dev, "QTI MHI QRTR driver probed\n");
|
||||
|
||||
return 0;
|
||||
|
|
|
@ -160,6 +160,7 @@ static struct work_struct qrtr_backup_work;
|
|||
* @kworker: worker thread for recv work
|
||||
* @task: task to run the worker thread
|
||||
* @read_data: scheduled work for recv work
|
||||
* @say_hello: scheduled work for initiating hello
|
||||
* @ws: wakeupsource avoid system suspend
|
||||
* @ilc: ipc logging context reference
|
||||
*/
|
||||
|
@ -170,6 +171,7 @@ struct qrtr_node {
|
|||
unsigned int nid;
|
||||
unsigned int net_id;
|
||||
atomic_t hello_sent;
|
||||
atomic_t hello_rcvd;
|
||||
|
||||
struct radix_tree_root qrtr_tx_flow;
|
||||
struct wait_queue_head resume_tx;
|
||||
|
@ -181,6 +183,7 @@ struct qrtr_node {
|
|||
struct kthread_worker kworker;
|
||||
struct task_struct *task;
|
||||
struct kthread_work read_data;
|
||||
struct kthread_work say_hello;
|
||||
|
||||
struct wakeup_source *ws;
|
||||
|
||||
|
@ -529,6 +532,10 @@ static int qrtr_node_enqueue(struct qrtr_node *node, struct sk_buff *skb,
|
|||
kfree_skb(skb);
|
||||
return rc;
|
||||
}
|
||||
if (atomic_read(&node->hello_sent) && type == QRTR_TYPE_HELLO) {
|
||||
kfree_skb(skb);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* If sk is null, this is a forwarded packet and should not wait */
|
||||
if (!skb->sk) {
|
||||
|
@ -966,6 +973,30 @@ static void qrtr_fwd_pkt(struct sk_buff *skb, struct qrtr_cb *cb)
|
|||
qrtr_node_enqueue(node, skb, cb->type, &from, &to, 0);
|
||||
qrtr_node_release(node);
|
||||
}
|
||||
|
||||
static void qrtr_sock_queue_skb(struct qrtr_node *node, struct sk_buff *skb,
|
||||
struct qrtr_sock *ipc)
|
||||
{
|
||||
struct qrtr_cb *cb = (struct qrtr_cb *)skb->cb;
|
||||
int rc;
|
||||
|
||||
/* Don't queue HELLO if control port already received */
|
||||
if (cb->type == QRTR_TYPE_HELLO) {
|
||||
if (atomic_read(&node->hello_rcvd)) {
|
||||
kfree_skb(skb);
|
||||
return;
|
||||
}
|
||||
atomic_inc(&node->hello_rcvd);
|
||||
}
|
||||
|
||||
rc = sock_queue_rcv_skb(&ipc->sk, skb);
|
||||
if (rc) {
|
||||
pr_err("%s: qrtr pkt dropped flow[%d] rc[%d]\n",
|
||||
__func__, cb->confirm_rx, rc);
|
||||
kfree_skb(skb);
|
||||
}
|
||||
}
|
||||
|
||||
/* Handle and route a received packet.
|
||||
*
|
||||
* This will auto-reply with resume-tx packet as necessary.
|
||||
|
@ -1008,18 +1039,40 @@ static void qrtr_node_rx_work(struct kthread_work *work)
|
|||
if (!ipc) {
|
||||
kfree_skb(skb);
|
||||
} else {
|
||||
if (sock_queue_rcv_skb(&ipc->sk, skb)) {
|
||||
pr_err("%s qrtr pkt dropped flow[%d]\n",
|
||||
__func__, cb->confirm_rx);
|
||||
kfree_skb(skb);
|
||||
}
|
||||
|
||||
qrtr_sock_queue_skb(node, skb, ipc);
|
||||
qrtr_port_put(ipc);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void qrtr_hello_work(struct kthread_work *work)
|
||||
{
|
||||
struct sockaddr_qrtr from = {AF_QIPCRTR, 0, QRTR_PORT_CTRL};
|
||||
struct sockaddr_qrtr to = {AF_QIPCRTR, 0, QRTR_PORT_CTRL};
|
||||
struct qrtr_ctrl_pkt *pkt;
|
||||
struct qrtr_node *node;
|
||||
struct qrtr_sock *ctrl;
|
||||
struct sk_buff *skb;
|
||||
|
||||
ctrl = qrtr_port_lookup(QRTR_PORT_CTRL);
|
||||
if (!ctrl)
|
||||
return;
|
||||
|
||||
skb = qrtr_alloc_ctrl_packet(&pkt);
|
||||
if (!skb) {
|
||||
qrtr_port_put(ctrl);
|
||||
return;
|
||||
}
|
||||
|
||||
node = container_of(work, struct qrtr_node, say_hello);
|
||||
pkt->cmd = cpu_to_le32(QRTR_TYPE_HELLO);
|
||||
from.sq_node = qrtr_local_nid;
|
||||
to.sq_node = node->nid;
|
||||
qrtr_node_enqueue(node, skb, QRTR_TYPE_HELLO, &from, &to, 0);
|
||||
qrtr_port_put(ctrl);
|
||||
}
|
||||
|
||||
/**
|
||||
* qrtr_endpoint_register() - register a new endpoint
|
||||
* @ep: endpoint to register
|
||||
|
@ -1048,8 +1101,10 @@ int qrtr_endpoint_register(struct qrtr_endpoint *ep, unsigned int net_id,
|
|||
node->nid = QRTR_EP_NID_AUTO;
|
||||
node->ep = ep;
|
||||
atomic_set(&node->hello_sent, 0);
|
||||
atomic_set(&node->hello_rcvd, 0);
|
||||
|
||||
kthread_init_work(&node->read_data, qrtr_node_rx_work);
|
||||
kthread_init_work(&node->say_hello, qrtr_hello_work);
|
||||
kthread_init_worker(&node->kworker);
|
||||
node->task = kthread_run(kthread_worker_fn, &node->kworker, "qrtr_rx");
|
||||
if (IS_ERR(node->task)) {
|
||||
|
@ -1071,6 +1126,7 @@ int qrtr_endpoint_register(struct qrtr_endpoint *ep, unsigned int net_id,
|
|||
up_write(&qrtr_node_lock);
|
||||
ep->node = node;
|
||||
|
||||
kthread_queue_work(&node->kworker, &node->say_hello);
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(qrtr_endpoint_register);
|
||||
|
@ -1352,6 +1408,17 @@ static int __qrtr_bind(struct socket *sock,
|
|||
qrtr_reset_ports();
|
||||
mutex_unlock(&qrtr_port_lock);
|
||||
|
||||
if (port == QRTR_PORT_CTRL) {
|
||||
struct qrtr_node *node;
|
||||
|
||||
down_write(&qrtr_node_lock);
|
||||
list_for_each_entry(node, &qrtr_all_epts, item) {
|
||||
atomic_set(&node->hello_sent, 0);
|
||||
atomic_set(&node->hello_rcvd, 0);
|
||||
}
|
||||
up_write(&qrtr_node_lock);
|
||||
}
|
||||
|
||||
/* unbind previous, if any */
|
||||
if (!zapped)
|
||||
qrtr_port_remove(ipc);
|
||||
|
@ -1451,7 +1518,7 @@ static int qrtr_bcast_enqueue(struct qrtr_node *node, struct sk_buff *skb,
|
|||
|
||||
down_read(&qrtr_node_lock);
|
||||
list_for_each_entry(node, &qrtr_all_epts, item) {
|
||||
if (node->nid == QRTR_EP_NID_AUTO)
|
||||
if (node->nid == QRTR_EP_NID_AUTO && type != QRTR_TYPE_HELLO)
|
||||
continue;
|
||||
skbn = skb_clone(skb, GFP_KERNEL);
|
||||
if (!skbn)
|
||||
|
|
Loading…
Reference in a new issue