int sctp_asconf_mgmt(struct sctp_sock *, struct sctp_sockaddr_entry *);
 struct sk_buff *sctp_skb_recv_datagram(struct sock *, int, int, int *);
 
+typedef int (*sctp_callback_t)(struct sctp_endpoint *, struct sctp_transport *, void *);
 void sctp_transport_walk_start(struct rhashtable_iter *iter);
 void sctp_transport_walk_stop(struct rhashtable_iter *iter);
 struct sctp_transport *sctp_transport_get_next(struct net *net,
                                  struct net *net,
                                  const union sctp_addr *laddr,
                                  const union sctp_addr *paddr, void *p);
-int sctp_for_each_transport(int (*cb)(struct sctp_transport *, void *),
-                           int (*cb_done)(struct sctp_transport *, void *),
-                           struct net *net, int *pos, void *p);
+int sctp_transport_traverse_process(sctp_callback_t cb, sctp_callback_t cb_done,
+                                   struct net *net, int *pos, void *p);
 int sctp_for_each_endpoint(int (*cb)(struct sctp_endpoint *, void *), void *p);
 int sctp_get_sctp_info(struct sock *sk, struct sctp_association *asoc,
                       struct sctp_info *info);
 
              reconf_enable:1;
 
        __u8  strreset_enable;
+       struct rcu_head rcu;
 };
 
 /* Recover the outter endpoint structure. */
 struct sctp_endpoint *sctp_endpoint_new(struct sock *, gfp_t);
 void sctp_endpoint_free(struct sctp_endpoint *);
 void sctp_endpoint_put(struct sctp_endpoint *);
-void sctp_endpoint_hold(struct sctp_endpoint *);
+int sctp_endpoint_hold(struct sctp_endpoint *ep);
 void sctp_endpoint_add_asoc(struct sctp_endpoint *, struct sctp_association *);
 struct sctp_association *sctp_endpoint_lookup_assoc(
        const struct sctp_endpoint *ep,
 
        return err;
 }
 
-static int sctp_sock_dump(struct sctp_transport *tsp, void *p)
+static int sctp_sock_dump(struct sctp_endpoint *ep, struct sctp_transport *tsp, void *p)
 {
-       struct sctp_endpoint *ep = tsp->asoc->ep;
        struct sctp_comm_param *commp = p;
        struct sock *sk = ep->base.sk;
        struct sk_buff *skb = commp->skb;
        int err = 0;
 
        lock_sock(sk);
+       if (ep != tsp->asoc->ep)
+               goto release;
        list_for_each_entry(assoc, &ep->asocs, asocs) {
                if (cb->args[4] < cb->args[1])
                        goto next;
        return err;
 }
 
-static int sctp_sock_filter(struct sctp_transport *tsp, void *p)
+static int sctp_sock_filter(struct sctp_endpoint *ep, struct sctp_transport *tsp, void *p)
 {
-       struct sctp_endpoint *ep = tsp->asoc->ep;
        struct sctp_comm_param *commp = p;
        struct sock *sk = ep->base.sk;
        const struct inet_diag_req_v2 *r = commp->r;
        if (!(idiag_states & ~(TCPF_LISTEN | TCPF_CLOSE)))
                goto done;
 
-       sctp_for_each_transport(sctp_sock_filter, sctp_sock_dump,
-                               net, &pos, &commp);
+       sctp_transport_traverse_process(sctp_sock_filter, sctp_sock_dump,
+                                       net, &pos, &commp);
        cb->args[2] = pos;
 
 done:
 
 }
 
 /* Final destructor for endpoint.  */
+static void sctp_endpoint_destroy_rcu(struct rcu_head *head)
+{
+       struct sctp_endpoint *ep = container_of(head, struct sctp_endpoint, rcu);
+       struct sock *sk = ep->base.sk;
+
+       sctp_sk(sk)->ep = NULL;
+       sock_put(sk);
+
+       kfree(ep);
+       SCTP_DBG_OBJCNT_DEC(ep);
+}
+
 static void sctp_endpoint_destroy(struct sctp_endpoint *ep)
 {
        struct sock *sk;
        if (sctp_sk(sk)->bind_hash)
                sctp_put_port(sk);
 
-       sctp_sk(sk)->ep = NULL;
-       /* Give up our hold on the sock */
-       sock_put(sk);
-
-       kfree(ep);
-       SCTP_DBG_OBJCNT_DEC(ep);
+       call_rcu(&ep->rcu, sctp_endpoint_destroy_rcu);
 }
 
 /* Hold a reference to an endpoint. */
-void sctp_endpoint_hold(struct sctp_endpoint *ep)
+int sctp_endpoint_hold(struct sctp_endpoint *ep)
 {
-       refcount_inc(&ep->base.refcnt);
+       return refcount_inc_not_zero(&ep->base.refcnt);
 }
 
 /* Release a reference to an endpoint and clean up if there are
 
 }
 EXPORT_SYMBOL_GPL(sctp_transport_lookup_process);
 
-int sctp_for_each_transport(int (*cb)(struct sctp_transport *, void *),
-                           int (*cb_done)(struct sctp_transport *, void *),
-                           struct net *net, int *pos, void *p) {
+int sctp_transport_traverse_process(sctp_callback_t cb, sctp_callback_t cb_done,
+                                   struct net *net, int *pos, void *p)
+{
        struct rhashtable_iter hti;
        struct sctp_transport *tsp;
+       struct sctp_endpoint *ep;
        int ret;
 
 again:
 
        tsp = sctp_transport_get_idx(net, &hti, *pos + 1);
        for (; !IS_ERR_OR_NULL(tsp); tsp = sctp_transport_get_next(net, &hti)) {
-               ret = cb(tsp, p);
-               if (ret)
-                       break;
+               ep = tsp->asoc->ep;
+               if (sctp_endpoint_hold(ep)) { /* asoc can be peeled off */
+                       ret = cb(ep, tsp, p);
+                       if (ret)
+                               break;
+                       sctp_endpoint_put(ep);
+               }
                (*pos)++;
                sctp_transport_put(tsp);
        }
        sctp_transport_walk_stop(&hti);
 
        if (ret) {
-               if (cb_done && !cb_done(tsp, p)) {
+               if (cb_done && !cb_done(ep, tsp, p)) {
                        (*pos)++;
+                       sctp_endpoint_put(ep);
                        sctp_transport_put(tsp);
                        goto again;
                }
+               sctp_endpoint_put(ep);
                sctp_transport_put(tsp);
        }
 
        return ret;
 }
-EXPORT_SYMBOL_GPL(sctp_for_each_transport);
+EXPORT_SYMBOL_GPL(sctp_transport_traverse_process);
 
 /* 7.2.1 Association Status (SCTP_STATUS)