Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 27503
b: refs/heads/master
c: bc0e646
h: refs/heads/master
i:
  27501: c1d5bad
  27499: 1390005
  27495: c152d14
  27487: fa6650e
v: v3
  • Loading branch information
Stephen Hemminger authored and David S. Miller committed Jun 18, 2006
1 parent 9a47b54 commit 68167ed
Show file tree
Hide file tree
Showing 3 changed files with 56 additions and 9 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: 8f182b494f87799d6ae20a1401825c516da46081
refs/heads/master: bc0e646796928918e45b6465e02616f2fe65c3c1
4 changes: 4 additions & 0 deletions trunk/include/net/llc_if.h
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,10 @@ static inline int llc_addrany(const struct llc_addr *addr)
return llc_mac_null(addr->mac) && !addr->lsap;
}

static inline int llc_mac_multicast(const u8 *mac)
{
return is_multicast_ether_addr(mac);
}
/**
* llc_mac_match - determines if two mac addresses are the same
* @mac1: First mac address to compare.
Expand Down
59 changes: 51 additions & 8 deletions trunk/net/llc/llc_sap.c
Original file line number Diff line number Diff line change
Expand Up @@ -282,7 +282,7 @@ static void llc_sap_rcv(struct llc_sap *sap, struct sk_buff *skb)
* mac, and local sap. Returns pointer for socket found, %NULL otherwise.
*/
static struct sock *llc_lookup_dgram(struct llc_sap *sap,
struct llc_addr *laddr)
const struct llc_addr *laddr)
{
struct sock *rc;
struct hlist_node *node;
Expand All @@ -304,19 +304,62 @@ static struct sock *llc_lookup_dgram(struct llc_sap *sap,
return rc;
}

/**
* llc_sap_mcast - Deliver multicast PDU's to all matching datagram sockets.
* @sap: SAP
* @laddr: address of local LLC (MAC + SAP)
*
* Search socket list of the SAP and finds connections with same sap.
* Deliver clone to each.
*/
static void llc_sap_mcast(struct llc_sap *sap,
const struct llc_addr *laddr,
struct sk_buff *skb)
{
struct sock *sk;
struct hlist_node *node;

read_lock_bh(&sap->sk_list.lock);
sk_for_each(sk, node, &sap->sk_list.list) {
struct llc_sock *llc = llc_sk(sk);
struct sk_buff *skb1;

if (sk->sk_type != SOCK_DGRAM)
continue;

if (llc->laddr.lsap != laddr->lsap)
continue;

skb1 = skb_clone(skb, GFP_ATOMIC);
if (!skb1)
break;

sock_hold(sk);
skb_set_owner_r(skb1, sk);
llc_sap_rcv(sap, skb1);
sock_put(sk);
}
read_unlock_bh(&sap->sk_list.lock);
}


void llc_sap_handler(struct llc_sap *sap, struct sk_buff *skb)
{
struct llc_addr laddr;
struct sock *sk;

llc_pdu_decode_da(skb, laddr.mac);
llc_pdu_decode_dsap(skb, &laddr.lsap);

sk = llc_lookup_dgram(sap, &laddr);
if (sk) {
skb_set_owner_r(skb, sk);
llc_sap_rcv(sap, skb);
sock_put(sk);
} else
if (llc_mac_multicast(laddr.mac)) {
llc_sap_mcast(sap, &laddr, skb);
kfree_skb(skb);
} else {
struct sock *sk = llc_lookup_dgram(sap, &laddr);
if (sk) {
skb_set_owner_r(skb, sk);
llc_sap_rcv(sap, skb);
sock_put(sk);
} else
kfree_skb(skb);
}
}

0 comments on commit 68167ed

Please sign in to comment.