2 Copyright (c) 2006, Intel Corporation
3 All rights reserved. This program and the accompanying materials
4 are licensed and made available under the terms and conditions of the BSD License
5 which accompanies this distribution. The full text of the license may be found at
6 http://opensource.org/licenses/bsd-license.php
8 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
9 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
17 2000-Feb-17 M(f)J Genesis.
24 pxe_rcvfilter_enable (
27 UINTN MCastAddressCount
,
28 EFI_MAC_ADDRESS
*MCastAddressList
33 this routine calls undi to enable the receive filters.
36 snp - pointer to snp driver structure
37 EnableFlags - bit mask for enabling the receive filters
38 MCastAddressCount - multicast address count for a new multicast address list
39 MCastAddressList - list of new multicast addresses
45 snp
->cdb
.OpCode
= PXE_OPCODE_RECEIVE_FILTERS
;
46 snp
->cdb
.OpFlags
= PXE_OPFLAGS_RECEIVE_FILTER_ENABLE
;
47 snp
->cdb
.CPBsize
= PXE_CPBSIZE_NOT_USED
;
48 snp
->cdb
.DBsize
= PXE_DBSIZE_NOT_USED
;
49 snp
->cdb
.CPBaddr
= PXE_CPBADDR_NOT_USED
;
50 snp
->cdb
.DBaddr
= PXE_DBADDR_NOT_USED
;
51 snp
->cdb
.StatCode
= PXE_STATCODE_INITIALIZE
;
52 snp
->cdb
.StatFlags
= PXE_STATFLAGS_INITIALIZE
;
53 snp
->cdb
.IFnum
= snp
->if_num
;
54 snp
->cdb
.Control
= PXE_CONTROL_LAST_CDB_IN_LIST
;
56 if ((EnableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_UNICAST
) != 0) {
57 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_UNICAST
;
60 if ((EnableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_BROADCAST
) != 0) {
61 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_BROADCAST
;
64 if ((EnableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_PROMISCUOUS
) != 0) {
65 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_PROMISCUOUS
;
68 if ((EnableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_PROMISCUOUS_MULTICAST
) != 0) {
69 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_ALL_MULTICAST
;
72 if ((EnableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_MULTICAST
) != 0) {
73 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_FILTERED_MULTICAST
;
76 if (MCastAddressCount
!= 0) {
77 snp
->cdb
.CPBsize
= (UINT16
) (MCastAddressCount
* sizeof (EFI_MAC_ADDRESS
));
78 snp
->cdb
.CPBaddr
= (UINT64
) (UINTN
) snp
->cpb
;
79 CopyMem (snp
->cpb
, MCastAddressList
, snp
->cdb
.CPBsize
);
82 // Issue UNDI command and check result.
84 DEBUG ((EFI_D_NET
, "\nsnp->undi.receive_filters() "));
86 (*snp
->issue_undi32_command
) ((UINT64
) (UINTN
) &snp
->cdb
);
88 if (snp
->cdb
.StatCode
!= EFI_SUCCESS
) {
90 // UNDI command failed. Return UNDI status to caller.
94 "\nsnp->undi.receive_filters() %xh:%xh\n",
99 switch (snp
->cdb
.StatCode
) {
100 case PXE_STATCODE_INVALID_CDB
:
101 case PXE_STATCODE_INVALID_CPB
:
102 case PXE_STATCODE_INVALID_PARAMETER
:
103 return EFI_INVALID_PARAMETER
;
105 case PXE_STATCODE_UNSUPPORTED
:
106 return EFI_UNSUPPORTED
;
109 return EFI_DEVICE_ERROR
;
116 pxe_rcvfilter_disable (
119 BOOLEAN ResetMCastList
124 this routine calls undi to disable the receive filters.
127 snp - pointer to snp driver structure
128 DisableFlags - bit mask for disabling the receive filters
129 ResetMCastList - boolean flag to reset/delete the multicast filter list
135 snp
->cdb
.OpCode
= PXE_OPCODE_RECEIVE_FILTERS
;
136 snp
->cdb
.CPBsize
= PXE_CPBSIZE_NOT_USED
;
137 snp
->cdb
.DBsize
= PXE_DBSIZE_NOT_USED
;
138 snp
->cdb
.CPBaddr
= PXE_CPBADDR_NOT_USED
;
139 snp
->cdb
.DBaddr
= PXE_DBADDR_NOT_USED
;
140 snp
->cdb
.StatCode
= PXE_STATCODE_INITIALIZE
;
141 snp
->cdb
.StatFlags
= PXE_STATFLAGS_INITIALIZE
;
142 snp
->cdb
.IFnum
= snp
->if_num
;
143 snp
->cdb
.Control
= PXE_CONTROL_LAST_CDB_IN_LIST
;
145 snp
->cdb
.OpFlags
= (UINT16
) (DisableFlags
? PXE_OPFLAGS_RECEIVE_FILTER_DISABLE
: PXE_OPFLAGS_NOT_USED
);
147 if (ResetMCastList
) {
148 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_RESET_MCAST_LIST
;
151 if ((DisableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_UNICAST
) != 0) {
152 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_UNICAST
;
155 if ((DisableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_BROADCAST
) != 0) {
156 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_BROADCAST
;
159 if ((DisableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_PROMISCUOUS
) != 0) {
160 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_PROMISCUOUS
;
163 if ((DisableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_PROMISCUOUS_MULTICAST
) != 0) {
164 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_ALL_MULTICAST
;
167 if ((DisableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_MULTICAST
) != 0) {
168 snp
->cdb
.OpFlags
|= PXE_OPFLAGS_RECEIVE_FILTER_FILTERED_MULTICAST
;
171 // Issue UNDI command and check result.
173 DEBUG ((EFI_D_NET
, "\nsnp->undi.receive_filters() "));
175 (*snp
->issue_undi32_command
) ((UINT64
) (UINTN
) &snp
->cdb
);
177 if (snp
->cdb
.StatCode
!= EFI_SUCCESS
) {
179 // UNDI command failed. Return UNDI status to caller.
183 "\nsnp->undi.receive_filters() %xh:%xh\n",
188 return EFI_DEVICE_ERROR
;
201 this routine calls undi to read the receive filters.
204 snp - pointer to snp driver structure
210 snp
->cdb
.OpCode
= PXE_OPCODE_RECEIVE_FILTERS
;
211 snp
->cdb
.OpFlags
= PXE_OPFLAGS_RECEIVE_FILTER_READ
;
212 snp
->cdb
.CPBsize
= PXE_CPBSIZE_NOT_USED
;
213 snp
->cdb
.DBsize
= (UINT16
) (snp
->mode
.MaxMCastFilterCount
* sizeof (EFI_MAC_ADDRESS
));
214 snp
->cdb
.CPBaddr
= PXE_CPBADDR_NOT_USED
;
215 if (snp
->cdb
.DBsize
== 0) {
216 snp
->cdb
.DBaddr
= (UINT64
) NULL
;
218 snp
->cdb
.DBaddr
= (UINT64
) (UINTN
) snp
->db
;
219 ZeroMem (snp
->db
, snp
->cdb
.DBsize
);
222 snp
->cdb
.StatCode
= PXE_STATCODE_INITIALIZE
;
223 snp
->cdb
.StatFlags
= PXE_STATFLAGS_INITIALIZE
;
224 snp
->cdb
.IFnum
= snp
->if_num
;
225 snp
->cdb
.Control
= PXE_CONTROL_LAST_CDB_IN_LIST
;
227 DEBUG ((EFI_D_NET
, "\nsnp->undi.receive_filters() "));
229 (*snp
->issue_undi32_command
) ((UINT64
) (UINTN
) &snp
->cdb
);
231 if (snp
->cdb
.StatCode
!= EFI_SUCCESS
) {
233 // UNDI command failed. Return UNDI status to caller.
237 "\nsnp->undi.receive_filters() %xh:%xh\n",
242 return EFI_DEVICE_ERROR
;
245 // Convert UNDI32 StatFlags to EFI SNP filter flags.
247 snp
->mode
.ReceiveFilterSetting
= 0;
249 if ((snp
->cdb
.StatFlags
& PXE_STATFLAGS_RECEIVE_FILTER_UNICAST
) != 0) {
250 snp
->mode
.ReceiveFilterSetting
|= EFI_SIMPLE_NETWORK_RECEIVE_UNICAST
;
253 if ((snp
->cdb
.StatFlags
& PXE_STATFLAGS_RECEIVE_FILTER_BROADCAST
) != 0) {
254 snp
->mode
.ReceiveFilterSetting
|= EFI_SIMPLE_NETWORK_RECEIVE_BROADCAST
;
257 if ((snp
->cdb
.StatFlags
& PXE_STATFLAGS_RECEIVE_FILTER_PROMISCUOUS
) != 0) {
258 snp
->mode
.ReceiveFilterSetting
|= EFI_SIMPLE_NETWORK_RECEIVE_PROMISCUOUS
;
261 if ((snp
->cdb
.StatFlags
& PXE_STATFLAGS_RECEIVE_FILTER_ALL_MULTICAST
) != 0) {
262 snp
->mode
.ReceiveFilterSetting
|= EFI_SIMPLE_NETWORK_RECEIVE_PROMISCUOUS_MULTICAST
;
265 if ((snp
->cdb
.StatFlags
& PXE_STATFLAGS_RECEIVE_FILTER_FILTERED_MULTICAST
) != 0) {
266 snp
->mode
.ReceiveFilterSetting
|= EFI_SIMPLE_NETWORK_RECEIVE_MULTICAST
;
269 CopyMem (snp
->mode
.MCastFilter
, snp
->db
, snp
->cdb
.DBsize
);
272 // Count number of active entries in multicast filter list.
275 EFI_MAC_ADDRESS ZeroMacAddr
;
277 SetMem (&ZeroMacAddr
, sizeof ZeroMacAddr
, 0);
279 for (snp
->mode
.MCastFilterCount
= 0;
280 snp
->mode
.MCastFilterCount
< snp
->mode
.MaxMCastFilterCount
;
281 snp
->mode
.MCastFilterCount
++
284 &snp
->mode
.MCastFilter
[snp
->mode
.MCastFilterCount
],
298 snp_undi32_receive_filters (
299 IN EFI_SIMPLE_NETWORK_PROTOCOL
* this,
300 IN UINT32 EnableFlags
,
301 IN UINT32 DisableFlags
,
302 IN BOOLEAN ResetMCastList
,
303 IN UINTN MCastAddressCount OPTIONAL
,
304 IN EFI_MAC_ADDRESS
* MCastAddressList OPTIONAL
309 This is the SNP interface routine for reading/enabling/disabling the
311 This routine basically retrieves snp structure, checks the SNP state and
312 checks the parameter validity, calls one of the above routines to actually
316 this - context pointer
317 EnableFlags - bit mask for enabling the receive filters
318 DisableFlags - bit mask for disabling the receive filters
319 ResetMCastList - boolean flag to reset/delete the multicast filter list
320 MCastAddressCount - multicast address count for a new multicast address list
321 MCastAddressList - list of new multicast addresses
331 return EFI_INVALID_PARAMETER
;
334 snp
= EFI_SIMPLE_NETWORK_DEV_FROM_THIS (this);
337 return EFI_DEVICE_ERROR
;
340 switch (snp
->mode
.State
) {
341 case EfiSimpleNetworkInitialized
:
344 case EfiSimpleNetworkStopped
:
345 return EFI_NOT_STARTED
;
347 case EfiSimpleNetworkStarted
:
348 return EFI_DEVICE_ERROR
;
351 return EFI_DEVICE_ERROR
;
354 // check if we are asked to enable or disable something that the UNDI
355 // does not even support!
357 if ((EnableFlags
&~snp
->mode
.ReceiveFilterMask
) != 0) {
358 return EFI_INVALID_PARAMETER
;
361 if ((DisableFlags
&~snp
->mode
.ReceiveFilterMask
) != 0) {
362 return EFI_INVALID_PARAMETER
;
365 if (ResetMCastList
) {
366 DisableFlags
|= EFI_SIMPLE_NETWORK_RECEIVE_MULTICAST
& snp
->mode
.ReceiveFilterMask
;
367 MCastAddressCount
= 0;
368 MCastAddressList
= NULL
;
370 if (MCastAddressCount
!= 0) {
371 if (MCastAddressCount
> snp
->mode
.MaxMCastFilterCount
) {
372 return EFI_INVALID_PARAMETER
;
375 if (MCastAddressList
== NULL
) {
376 return EFI_INVALID_PARAMETER
;
381 if (EnableFlags
== 0 && DisableFlags
== 0 && !ResetMCastList
&& MCastAddressCount
== 0) {
385 if ((EnableFlags
& EFI_SIMPLE_NETWORK_RECEIVE_MULTICAST
) != 0 && MCastAddressCount
== 0) {
386 return EFI_INVALID_PARAMETER
;
389 if ((EnableFlags
!= 0) || (MCastAddressCount
!= 0)) {
390 Status
= pxe_rcvfilter_enable (
397 if (Status
!= EFI_SUCCESS
) {
402 if ((DisableFlags
!= 0) || ResetMCastList
) {
403 Status
= pxe_rcvfilter_disable (snp
, DisableFlags
, ResetMCastList
);
405 if (Status
!= EFI_SUCCESS
) {
410 return pxe_rcvfilter_read (snp
);