VirtualBox

source: vbox/trunk/src/VBox/NetworkServices/NetLib/VBoxNetARP.cpp@ 18483

Last change on this file since 18483 was 17783, checked in by vboxsync, 16 years ago

ARP.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.1 KB
Line 
1/* $Id: VBoxNetARP.cpp 17783 2009-03-12 23:59:57Z vboxsync $ */
2/** @file
3 * VBoxNetARP - IntNet ARP Client Routines.
4 */
5
6/*
7 * Copyright (C) 2009 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22/*******************************************************************************
23* Header Files *
24*******************************************************************************/
25#define LOG_GROUP LOG_GROUP_DEFAULT
26#include "VBoxNetLib.h"
27#include <iprt/string.h>
28#include <VBox/log.h>
29
30
31/**
32 * Deal with ARP queries.
33 *
34 * @returns true if ARP.
35 *
36 * @param pSession The support driver session.
37 * @param hIf The internal network interface handle.
38 * @param pBuf The internal network interface buffer.
39 * @param pMacAddr Our MAC address.
40 * @param IPv4Addr Our IPv4 address.
41 */
42bool VBoxNetArpHandleIt(PSUPDRVSESSION pSession, INTNETIFHANDLE hIf, PINTNETBUF pBuf, PCRTMAC pMacAddr, RTNETADDRIPV4 IPv4Addr)
43{
44 /*
45 * Valid IntNet Ethernet frame?
46 */
47 PCINTNETHDR pHdr = (PINTNETHDR)((uintptr_t)pBuf + pBuf->Recv.offRead);
48 if (pHdr->u16Type != INTNETHDR_TYPE_FRAME)
49 return false;
50
51 size_t cbFrame = pHdr->cbFrame;
52 const void *pvFrame = INTNETHdrGetFramePtr(pHdr, pBuf);
53 PCRTNETETHERHDR pEthHdr = (PCRTNETETHERHDR)pvFrame;
54
55 /*
56 * Arp frame?
57 */
58 if (pEthHdr->EtherType != RT_H2N_U16_C(RTNET_ETHERTYPE_ARP))
59 return false;
60 if ( ( pEthHdr->DstMac.au16[0] != 0xffff
61 || pEthHdr->DstMac.au16[1] != 0xffff
62 || pEthHdr->DstMac.au16[2] != 0xffff)
63 && ( pEthHdr->DstMac.au16[0] != pMacAddr->au16[0]
64 || pEthHdr->DstMac.au16[1] != pMacAddr->au16[1]
65 || pEthHdr->DstMac.au16[2] != pMacAddr->au16[2])
66 )
67 return false;
68 if (cbFrame < sizeof(RTNETARPIPV4) + sizeof(RTNETETHERHDR))
69 return false;
70
71 PCRTNETARPHDR pArpHdr = (PCRTNETARPHDR)(pEthHdr + 1);
72 if (pArpHdr->ar_htype != RT_H2N_U16_C(RTNET_ARP_ETHER))
73 return false;
74 if (pArpHdr->ar_hlen != sizeof(RTMAC))
75 return false;
76 if (pArpHdr->ar_ptype != RT_H2N_U16_C(RTNET_ETHERTYPE_IPV4))
77 return false;
78 if (pArpHdr->ar_plen != sizeof(RTNETADDRIPV4))
79 return false;
80
81 /* It's ARP, alright. Anything we need to do something about. */
82 PCRTNETARPIPV4 pArp = (PCRTNETARPIPV4)pArpHdr;
83 switch (pArp->Hdr.ar_oper)
84 {
85 case RT_H2N_U16_C(RTNET_ARPOP_REQUEST):
86 case RT_H2N_U16_C(RTNET_ARPOP_REVREQUEST):
87 case RT_H2N_U16_C(RTNET_ARPOP_INVREQUEST):
88 break;
89 default:
90 return true;
91 }
92
93 /*
94 * Deal with the queries.
95 */
96 RTNETARPIPV4 Reply;
97 switch (pArp->Hdr.ar_oper)
98 {
99 /* 'Who has ar_tpa? Tell ar_spa.' */
100 case RT_H2N_U16_C(RTNET_ARPOP_REQUEST):
101 if (pArp->ar_tpa.u != IPv4Addr.u)
102 return true;
103 Reply.Hdr.ar_oper = RT_H2N_U16_C(RTNET_ARPOP_REPLY);
104 break;
105
106 case RT_H2N_U16_C(RTNET_ARPOP_REVREQUEST):
107 if ( pArp->ar_tha.au16[0] != pMacAddr->au16[0]
108 || pArp->ar_tha.au16[1] != pMacAddr->au16[1]
109 || pArp->ar_tha.au16[2] != pMacAddr->au16[2])
110 return true;
111 Reply.Hdr.ar_oper = RT_H2N_U16_C(RTNET_ARPOP_REVREPLY);
112 break;
113
114 case RT_H2N_U16_C(RTNET_ARPOP_INVREQUEST):
115 /** @todo RTNET_ARPOP_INVREQUEST */
116 return true;
117 //Reply.Hdr.ar_oper = RT_H2N_U16_C(RTNET_ARPOP_INVREPLY);
118 //break;
119 }
120
121 /*
122 * Complete the reply and send it.
123 */
124 Reply.Hdr.ar_htype = RT_H2N_U16_C(RTNET_ARP_ETHER);
125 Reply.Hdr.ar_ptype = RT_H2N_U16_C(RTNET_ETHERTYPE_IPV4);
126 Reply.Hdr.ar_hlen = sizeof(RTMAC);
127 Reply.Hdr.ar_plen = sizeof(RTNETADDRIPV4);
128 Reply.ar_sha = *pMacAddr;
129 Reply.ar_spa = IPv4Addr;
130 Reply.ar_tha = pArp->ar_sha;
131 Reply.ar_tpa = pArp->ar_spa;
132
133
134 RTNETETHERHDR EthHdr;
135 EthHdr.DstMac = pArp->ar_sha;
136 EthHdr.SrcMac = *pMacAddr;
137 EthHdr.EtherType = RT_H2N_U16_C(RTNET_ETHERTYPE_ARP);
138
139 uint8_t abTrailer[60 - sizeof(Reply) - sizeof(EthHdr)];
140 memset(abTrailer, '\0', sizeof(abTrailer));
141
142 INTNETSEG aSegs[3];
143 aSegs[0].cb = sizeof(EthHdr);
144 aSegs[0].pv = &EthHdr;
145
146 aSegs[1].pv = &Reply;
147 aSegs[1].cb = sizeof(Reply);
148
149 aSegs[2].pv = &abTrailer[0];
150 aSegs[2].cb = sizeof(abTrailer);
151
152 VBoxNetIntIfSend(pSession, hIf, pBuf, RT_ELEMENTS(aSegs), &aSegs[0], true /* fFlush */);
153
154 return true;
155}
156
157
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette