Merge tag 'drm-intel-next-fixes-2016-05-25' of git://anongit.freedesktop.org/drm...
[deliverable/linux.git] / net / sctp / socket.c
index 878d28eda1a68dc639d7b9f3f663d7518f21bb32..777d0324594a33a407e9ec157a7634334b1292e2 100644 (file)
@@ -4202,6 +4202,222 @@ static void sctp_shutdown(struct sock *sk, int how)
        }
 }
 
+int sctp_get_sctp_info(struct sock *sk, struct sctp_association *asoc,
+                      struct sctp_info *info)
+{
+       struct sctp_transport *prim;
+       struct list_head *pos;
+       int mask;
+
+       memset(info, 0, sizeof(*info));
+       if (!asoc) {
+               struct sctp_sock *sp = sctp_sk(sk);
+
+               info->sctpi_s_autoclose = sp->autoclose;
+               info->sctpi_s_adaptation_ind = sp->adaptation_ind;
+               info->sctpi_s_pd_point = sp->pd_point;
+               info->sctpi_s_nodelay = sp->nodelay;
+               info->sctpi_s_disable_fragments = sp->disable_fragments;
+               info->sctpi_s_v4mapped = sp->v4mapped;
+               info->sctpi_s_frag_interleave = sp->frag_interleave;
+
+               return 0;
+       }
+
+       info->sctpi_tag = asoc->c.my_vtag;
+       info->sctpi_state = asoc->state;
+       info->sctpi_rwnd = asoc->a_rwnd;
+       info->sctpi_unackdata = asoc->unack_data;
+       info->sctpi_penddata = sctp_tsnmap_pending(&asoc->peer.tsn_map);
+       info->sctpi_instrms = asoc->c.sinit_max_instreams;
+       info->sctpi_outstrms = asoc->c.sinit_num_ostreams;
+       list_for_each(pos, &asoc->base.inqueue.in_chunk_list)
+               info->sctpi_inqueue++;
+       list_for_each(pos, &asoc->outqueue.out_chunk_list)
+               info->sctpi_outqueue++;
+       info->sctpi_overall_error = asoc->overall_error_count;
+       info->sctpi_max_burst = asoc->max_burst;
+       info->sctpi_maxseg = asoc->frag_point;
+       info->sctpi_peer_rwnd = asoc->peer.rwnd;
+       info->sctpi_peer_tag = asoc->c.peer_vtag;
+
+       mask = asoc->peer.ecn_capable << 1;
+       mask = (mask | asoc->peer.ipv4_address) << 1;
+       mask = (mask | asoc->peer.ipv6_address) << 1;
+       mask = (mask | asoc->peer.hostname_address) << 1;
+       mask = (mask | asoc->peer.asconf_capable) << 1;
+       mask = (mask | asoc->peer.prsctp_capable) << 1;
+       mask = (mask | asoc->peer.auth_capable);
+       info->sctpi_peer_capable = mask;
+       mask = asoc->peer.sack_needed << 1;
+       mask = (mask | asoc->peer.sack_generation) << 1;
+       mask = (mask | asoc->peer.zero_window_announced);
+       info->sctpi_peer_sack = mask;
+
+       info->sctpi_isacks = asoc->stats.isacks;
+       info->sctpi_osacks = asoc->stats.osacks;
+       info->sctpi_opackets = asoc->stats.opackets;
+       info->sctpi_ipackets = asoc->stats.ipackets;
+       info->sctpi_rtxchunks = asoc->stats.rtxchunks;
+       info->sctpi_outofseqtsns = asoc->stats.outofseqtsns;
+       info->sctpi_idupchunks = asoc->stats.idupchunks;
+       info->sctpi_gapcnt = asoc->stats.gapcnt;
+       info->sctpi_ouodchunks = asoc->stats.ouodchunks;
+       info->sctpi_iuodchunks = asoc->stats.iuodchunks;
+       info->sctpi_oodchunks = asoc->stats.oodchunks;
+       info->sctpi_iodchunks = asoc->stats.iodchunks;
+       info->sctpi_octrlchunks = asoc->stats.octrlchunks;
+       info->sctpi_ictrlchunks = asoc->stats.ictrlchunks;
+
+       prim = asoc->peer.primary_path;
+       memcpy(&info->sctpi_p_address, &prim->ipaddr,
+              sizeof(struct sockaddr_storage));
+       info->sctpi_p_state = prim->state;
+       info->sctpi_p_cwnd = prim->cwnd;
+       info->sctpi_p_srtt = prim->srtt;
+       info->sctpi_p_rto = jiffies_to_msecs(prim->rto);
+       info->sctpi_p_hbinterval = prim->hbinterval;
+       info->sctpi_p_pathmaxrxt = prim->pathmaxrxt;
+       info->sctpi_p_sackdelay = jiffies_to_msecs(prim->sackdelay);
+       info->sctpi_p_ssthresh = prim->ssthresh;
+       info->sctpi_p_partial_bytes_acked = prim->partial_bytes_acked;
+       info->sctpi_p_flight_size = prim->flight_size;
+       info->sctpi_p_error = prim->error_count;
+
+       return 0;
+}
+EXPORT_SYMBOL_GPL(sctp_get_sctp_info);
+
+/* use callback to avoid exporting the core structure */
+int sctp_transport_walk_start(struct rhashtable_iter *iter)
+{
+       int err;
+
+       err = rhashtable_walk_init(&sctp_transport_hashtable, iter,
+                                  GFP_KERNEL);
+       if (err)
+               return err;
+
+       err = rhashtable_walk_start(iter);
+       if (err && err != -EAGAIN) {
+               rhashtable_walk_exit(iter);
+               return err;
+       }
+
+       return 0;
+}
+
+void sctp_transport_walk_stop(struct rhashtable_iter *iter)
+{
+       rhashtable_walk_stop(iter);
+       rhashtable_walk_exit(iter);
+}
+
+struct sctp_transport *sctp_transport_get_next(struct net *net,
+                                              struct rhashtable_iter *iter)
+{
+       struct sctp_transport *t;
+
+       t = rhashtable_walk_next(iter);
+       for (; t; t = rhashtable_walk_next(iter)) {
+               if (IS_ERR(t)) {
+                       if (PTR_ERR(t) == -EAGAIN)
+                               continue;
+                       break;
+               }
+
+               if (net_eq(sock_net(t->asoc->base.sk), net) &&
+                   t->asoc->peer.primary_path == t)
+                       break;
+       }
+
+       return t;
+}
+
+struct sctp_transport *sctp_transport_get_idx(struct net *net,
+                                             struct rhashtable_iter *iter,
+                                             int pos)
+{
+       void *obj = SEQ_START_TOKEN;
+
+       while (pos && (obj = sctp_transport_get_next(net, iter)) &&
+              !IS_ERR(obj))
+               pos--;
+
+       return obj;
+}
+
+int sctp_for_each_endpoint(int (*cb)(struct sctp_endpoint *, void *),
+                          void *p) {
+       int err = 0;
+       int hash = 0;
+       struct sctp_ep_common *epb;
+       struct sctp_hashbucket *head;
+
+       for (head = sctp_ep_hashtable; hash < sctp_ep_hashsize;
+            hash++, head++) {
+               read_lock(&head->lock);
+               sctp_for_each_hentry(epb, &head->chain) {
+                       err = cb(sctp_ep(epb), p);
+                       if (err)
+                               break;
+               }
+               read_unlock(&head->lock);
+       }
+
+       return err;
+}
+EXPORT_SYMBOL_GPL(sctp_for_each_endpoint);
+
+int sctp_transport_lookup_process(int (*cb)(struct sctp_transport *, void *),
+                                 struct net *net,
+                                 const union sctp_addr *laddr,
+                                 const union sctp_addr *paddr, void *p)
+{
+       struct sctp_transport *transport;
+       int err = 0;
+
+       rcu_read_lock();
+       transport = sctp_addrs_lookup_transport(net, laddr, paddr);
+       if (!transport || !sctp_transport_hold(transport))
+               goto out;
+       err = cb(transport, p);
+       sctp_transport_put(transport);
+
+out:
+       rcu_read_unlock();
+       return err;
+}
+EXPORT_SYMBOL_GPL(sctp_transport_lookup_process);
+
+int sctp_for_each_transport(int (*cb)(struct sctp_transport *, void *),
+                           struct net *net, int pos, void *p) {
+       struct rhashtable_iter hti;
+       void *obj;
+       int err;
+
+       err = sctp_transport_walk_start(&hti);
+       if (err)
+               return err;
+
+       sctp_transport_get_idx(net, &hti, pos);
+       obj = sctp_transport_get_next(net, &hti);
+       for (; obj && !IS_ERR(obj); obj = sctp_transport_get_next(net, &hti)) {
+               struct sctp_transport *transport = obj;
+
+               if (!sctp_transport_hold(transport))
+                       continue;
+               err = cb(transport, p);
+               sctp_transport_put(transport);
+               if (err)
+                       break;
+       }
+       sctp_transport_walk_stop(&hti);
+
+       return err;
+}
+EXPORT_SYMBOL_GPL(sctp_for_each_transport);
+
 /* 7.2.1 Association Status (SCTP_STATUS)
 
  * Applications can retrieve current status information about an
@@ -6430,6 +6646,8 @@ unsigned int sctp_poll(struct file *file, struct socket *sock, poll_table *wait)
 
        poll_wait(file, sk_sleep(sk), wait);
 
+       sock_rps_record_flow(sk);
+
        /* A TCP-style listening socket becomes readable when the accept queue
         * is not empty.
         */
@@ -6764,13 +6982,11 @@ struct sk_buff *sctp_skb_recv_datagram(struct sock *sk, int flags,
                 *  However, this function was correct in any case. 8)
                 */
                if (flags & MSG_PEEK) {
-                       spin_lock_bh(&sk->sk_receive_queue.lock);
                        skb = skb_peek(&sk->sk_receive_queue);
                        if (skb)
                                atomic_inc(&skb->users);
-                       spin_unlock_bh(&sk->sk_receive_queue.lock);
                } else {
-                       skb = skb_dequeue(&sk->sk_receive_queue);
+                       skb = __skb_dequeue(&sk->sk_receive_queue);
                }
 
                if (skb)
@@ -7186,6 +7402,7 @@ void sctp_copy_sock(struct sock *newsk, struct sock *sk,
        newsk->sk_lingertime = sk->sk_lingertime;
        newsk->sk_rcvtimeo = sk->sk_rcvtimeo;
        newsk->sk_sndtimeo = sk->sk_sndtimeo;
+       newsk->sk_rxhash = sk->sk_rxhash;
 
        newinet = inet_sk(newsk);
 
This page took 0.026661 seconds and 5 git commands to generate.