summaryrefslogtreecommitdiff
path: root/accel-pppd/ctrl/ipoe/arp.c
diff options
context:
space:
mode:
Diffstat (limited to 'accel-pppd/ctrl/ipoe/arp.c')
-rw-r--r--accel-pppd/ctrl/ipoe/arp.c33
1 files changed, 28 insertions, 5 deletions
diff --git a/accel-pppd/ctrl/ipoe/arp.c b/accel-pppd/ctrl/ipoe/arp.c
index 58533582..36b03447 100644
--- a/accel-pppd/ctrl/ipoe/arp.c
+++ b/accel-pppd/ctrl/ipoe/arp.c
@@ -39,6 +39,8 @@ struct arp_tree {
static mempool_t arp_pool;
static mempool_t arp_hdr_pool;
+static uint8_t bc_addr[ETH_ALEN] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
+
#define HASH_BITS 0xff
static struct arp_tree *arp_tree;
@@ -63,6 +65,12 @@ static void arp_ctx_read(struct _arphdr *ah)
ah2.ar_op = htons(ARPOP_REPLY);
pthread_mutex_lock(&ipoe->lock);
+ if (ah->ar_op == htons(ARPOP_REPLY)) {
+ ipoe_serv_recv_arp(ipoe, ah);
+ pthread_mutex_unlock(&ipoe->lock);
+ goto out;
+ }
+
list_for_each_entry(ses, &ipoe->sessions, entry) {
if (ses->yiaddr == ah->ar_spa) {
ses1 = ses;
@@ -80,8 +88,14 @@ static void arp_ctx_read(struct _arphdr *ah)
break;
}
- if (!ses1 || (ses1->ses.state != AP_STATE_ACTIVE) ||
- (ses2 && ses2->ses.state != AP_STATE_ACTIVE)) {
+ if (!ses1 && ipoe->opt_up) {
+ ipoe_serv_recv_arp(ipoe, ah);
+ pthread_mutex_unlock(&ipoe->lock);
+ goto out;
+ }
+
+ if (!ipoe->opt_arp || !ses1 || ses1->arph ||
+ (ses2 && ses2->ses.state != AP_STATE_ACTIVE)) {
pthread_mutex_unlock(&ipoe->lock);
goto out;
}
@@ -112,7 +126,7 @@ out:
mempool_free(ah);
}
-void arp_send(int ifindex, struct _arphdr *arph)
+void arp_send(int ifindex, struct _arphdr *arph, int broadcast)
{
struct sockaddr_ll dst;
@@ -120,6 +134,10 @@ void arp_send(int ifindex, struct _arphdr *arph)
dst.sll_family = AF_PACKET;
dst.sll_ifindex = ifindex;
dst.sll_protocol = htons(ETH_P_ARP);
+ if (broadcast)
+ memcpy(dst.sll_addr, bc_addr, ETH_ALEN);
+ else
+ memcpy(dst.sll_addr, arph->ar_tha, ETH_ALEN);
arph->ar_op = htons(ARPOP_REPLY);
@@ -150,8 +168,13 @@ static int arp_read(struct triton_md_handler_t *h)
if (r < sizeof(*ah))
continue;
- if (ah->ar_op != htons(ARPOP_REQUEST))
- continue;
+ if (ah->ar_op != htons(ARPOP_REQUEST)) {
+ if (ah->ar_op != htons(ARPOP_REPLY))
+ continue;
+
+ if (memcmp(src.sll_addr, bc_addr, ETH_ALEN))
+ continue;
+ }
if (ah->ar_pln != 4)
continue;