• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /** @file
2     Implementation of transmitting a packet.
3 
4 Copyright (c) 2004 - 2016, Intel Corporation. All rights reserved.<BR>
5 This program and the accompanying materials are licensed
6 and made available under the terms and conditions of the BSD License which
7 accompanies this distribution. The full text of the license may be found at
8 http://opensource.org/licenses/bsd-license.php
9 
10 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
11 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
12 
13 **/
14 
15 #include "Snp.h"
16 
17 
18 /**
19   Call UNDI to create the meadia header for the given data buffer.
20 
21   @param  Snp              Pointer to SNP driver structure.
22   @param  MacHeaderPtr     Address where the media header will be filled in.
23   @param  HeaderSize       Size of the memory at MacHeaderPtr.
24   @param  Buffer           Data buffer pointer.
25   @param  BufferSize       Size of data in the Buffer
26   @param  DestAddr         Address of the destination mac address buffer.
27   @param  SrcAddr          Address of the source mac address buffer.
28   @param  ProtocolPtr      Address of the protocol type.
29 
30   @retval EFI_SUCCESS      Successfully completed the undi call.
31   @retval Other            Error return from undi call.
32 
33 **/
34 EFI_STATUS
PxeFillHeader(SNP_DRIVER * Snp,VOID * MacHeaderPtr,UINTN HeaderSize,VOID * Buffer,UINTN BufferSize,EFI_MAC_ADDRESS * DestAddr,EFI_MAC_ADDRESS * SrcAddr,UINT16 * ProtocolPtr)35 PxeFillHeader (
36   SNP_DRIVER      *Snp,
37   VOID            *MacHeaderPtr,
38   UINTN           HeaderSize,
39   VOID            *Buffer,
40   UINTN           BufferSize,
41   EFI_MAC_ADDRESS *DestAddr,
42   EFI_MAC_ADDRESS *SrcAddr,
43   UINT16          *ProtocolPtr
44   )
45 {
46   PXE_CPB_FILL_HEADER_FRAGMENTED  *Cpb;
47 
48   Cpb = Snp->Cpb;
49   if (SrcAddr != NULL) {
50     CopyMem (
51       (VOID *) Cpb->SrcAddr,
52       (VOID *) SrcAddr,
53       Snp->Mode.HwAddressSize
54       );
55   } else {
56     CopyMem (
57       (VOID *) Cpb->SrcAddr,
58       (VOID *) &(Snp->Mode.CurrentAddress),
59       Snp->Mode.HwAddressSize
60       );
61   }
62 
63   CopyMem (
64     (VOID *) Cpb->DestAddr,
65     (VOID *) DestAddr,
66     Snp->Mode.HwAddressSize
67     );
68 
69   //
70   // we need to do the byte swapping
71   //
72   Cpb->Protocol             = (UINT16) PXE_SWAP_UINT16 (*ProtocolPtr);
73 
74   Cpb->PacketLen            = (UINT32) (BufferSize);
75   Cpb->MediaHeaderLen       = (UINT16) HeaderSize;
76 
77   Cpb->FragCnt              = 2;
78   Cpb->reserved             = 0;
79 
80   Cpb->FragDesc[0].FragAddr = (UINT64)(UINTN) MacHeaderPtr;
81   Cpb->FragDesc[0].FragLen  = (UINT32) HeaderSize;
82   Cpb->FragDesc[1].FragAddr = (UINT64)(UINTN) Buffer;
83   Cpb->FragDesc[1].FragLen  = (UINT32) BufferSize;
84 
85   Cpb->FragDesc[0].reserved = Cpb->FragDesc[1].reserved = 0;
86 
87   Snp->Cdb.OpCode     = PXE_OPCODE_FILL_HEADER;
88   Snp->Cdb.OpFlags    = PXE_OPFLAGS_FILL_HEADER_FRAGMENTED;
89 
90   Snp->Cdb.DBsize     = PXE_DBSIZE_NOT_USED;
91   Snp->Cdb.DBaddr     = PXE_DBADDR_NOT_USED;
92 
93   Snp->Cdb.CPBsize    = (UINT16) sizeof (PXE_CPB_FILL_HEADER_FRAGMENTED);
94   Snp->Cdb.CPBaddr    = (UINT64)(UINTN) Cpb;
95 
96   Snp->Cdb.StatCode   = PXE_STATCODE_INITIALIZE;
97   Snp->Cdb.StatFlags  = PXE_STATFLAGS_INITIALIZE;
98   Snp->Cdb.IFnum      = Snp->IfNum;
99   Snp->Cdb.Control    = PXE_CONTROL_LAST_CDB_IN_LIST;
100 
101   //
102   // Issue UNDI command and check result.
103   //
104   DEBUG ((EFI_D_NET, "\nSnp->undi.fill_header()  "));
105 
106   (*Snp->IssueUndi32Command) ((UINT64) (UINTN) &Snp->Cdb);
107 
108   switch (Snp->Cdb.StatCode) {
109   case PXE_STATCODE_SUCCESS:
110     return EFI_SUCCESS;
111 
112   case PXE_STATCODE_INVALID_PARAMETER:
113     DEBUG (
114       (EFI_D_ERROR,
115       "\nSnp->undi.fill_header()  %xh:%xh\n",
116       Snp->Cdb.StatFlags,
117       Snp->Cdb.StatCode)
118       );
119 
120     return EFI_INVALID_PARAMETER;
121 
122   default:
123     DEBUG (
124       (EFI_D_ERROR,
125       "\nSnp->undi.fill_header()  %xh:%xh\n",
126       Snp->Cdb.StatFlags,
127       Snp->Cdb.StatCode)
128       );
129 
130     return EFI_DEVICE_ERROR;
131   }
132 }
133 
134 
135 /**
136   This routine calls undi to transmit the given data buffer
137 
138   @param  Snp                 pointer to SNP driver structure
139   @param  Buffer           data buffer pointer
140   @param  BufferSize        Size of data in the Buffer
141 
142   @retval EFI_SUCCESS         if successfully completed the undi call
143   @retval Other               error return from undi call.
144 
145 **/
146 EFI_STATUS
PxeTransmit(SNP_DRIVER * Snp,VOID * Buffer,UINTN BufferSize)147 PxeTransmit (
148   SNP_DRIVER *Snp,
149   VOID       *Buffer,
150   UINTN      BufferSize
151   )
152 {
153   PXE_CPB_TRANSMIT  *Cpb;
154   EFI_STATUS        Status;
155 
156   Cpb             = Snp->Cpb;
157   Cpb->FrameAddr  = (UINT64) (UINTN) Buffer;
158   Cpb->DataLen    = (UINT32) BufferSize;
159 
160   Cpb->MediaheaderLen = 0;
161   Cpb->reserved       = 0;
162 
163   Snp->Cdb.OpFlags    = PXE_OPFLAGS_TRANSMIT_WHOLE;
164 
165   Snp->Cdb.CPBsize    = (UINT16) sizeof (PXE_CPB_TRANSMIT);
166   Snp->Cdb.CPBaddr    = (UINT64)(UINTN) Cpb;
167 
168   Snp->Cdb.OpCode     = PXE_OPCODE_TRANSMIT;
169   Snp->Cdb.DBsize     = PXE_DBSIZE_NOT_USED;
170   Snp->Cdb.DBaddr     = PXE_DBADDR_NOT_USED;
171 
172   Snp->Cdb.StatCode   = PXE_STATCODE_INITIALIZE;
173   Snp->Cdb.StatFlags  = PXE_STATFLAGS_INITIALIZE;
174   Snp->Cdb.IFnum      = Snp->IfNum;
175   Snp->Cdb.Control    = PXE_CONTROL_LAST_CDB_IN_LIST;
176 
177   //
178   // Issue UNDI command and check result.
179   //
180   DEBUG ((EFI_D_NET, "\nSnp->undi.transmit()  "));
181   DEBUG ((EFI_D_NET, "\nSnp->Cdb.OpCode  == %x", Snp->Cdb.OpCode));
182   DEBUG ((EFI_D_NET, "\nSnp->Cdb.CPBaddr == %LX", Snp->Cdb.CPBaddr));
183   DEBUG ((EFI_D_NET, "\nSnp->Cdb.DBaddr  == %LX", Snp->Cdb.DBaddr));
184   DEBUG ((EFI_D_NET, "\nCpb->FrameAddr   == %LX\n", Cpb->FrameAddr));
185 
186   (*Snp->IssueUndi32Command) ((UINT64) (UINTN) &Snp->Cdb);
187 
188   DEBUG ((EFI_D_NET, "\nexit Snp->undi.transmit()  "));
189   DEBUG ((EFI_D_NET, "\nSnp->Cdb.StatCode == %r", Snp->Cdb.StatCode));
190 
191   //
192   // we will unmap the buffers in get_status call, not here
193   //
194   switch (Snp->Cdb.StatCode) {
195   case PXE_STATCODE_SUCCESS:
196     return EFI_SUCCESS;
197 
198   case PXE_STATCODE_BUFFER_FULL:
199   case PXE_STATCODE_QUEUE_FULL:
200   case PXE_STATCODE_BUSY:
201     Status = EFI_NOT_READY;
202     break;
203 
204   default:
205     Status = EFI_DEVICE_ERROR;
206   }
207 
208   DEBUG (
209     (EFI_D_ERROR,
210     "\nSnp->undi.transmit()  %xh:%xh\n",
211     Snp->Cdb.StatFlags,
212     Snp->Cdb.StatCode)
213     );
214 
215   return Status;
216 }
217 
218 /**
219   Places a packet in the transmit queue of a network interface.
220 
221   This function places the packet specified by Header and Buffer on the transmit
222   queue. If HeaderSize is nonzero and HeaderSize is not equal to
223   This->Mode->MediaHeaderSize, then EFI_INVALID_PARAMETER will be returned. If
224   BufferSize is less than This->Mode->MediaHeaderSize, then EFI_BUFFER_TOO_SMALL
225   will be returned. If Buffer is NULL, then EFI_INVALID_PARAMETER will be
226   returned. If HeaderSize is nonzero and DestAddr or Protocol is NULL, then
227   EFI_INVALID_PARAMETER will be returned. If the transmit engine of the network
228   interface is busy, then EFI_NOT_READY will be returned. If this packet can be
229   accepted by the transmit engine of the network interface, the packet contents
230   specified by Buffer will be placed on the transmit queue of the network
231   interface, and EFI_SUCCESS will be returned. GetStatus() can be used to
232   determine when the packet has actually been transmitted. The contents of the
233   Buffer must not be modified until the packet has actually been transmitted.
234   The Transmit() function performs nonblocking I/O. A caller who wants to perform
235   blocking I/O, should call Transmit(), and then GetStatus() until the
236   transmitted buffer shows up in the recycled transmit buffer.
237   If the driver has not been initialized, EFI_DEVICE_ERROR will be returned.
238 
239   @param This       A pointer to the EFI_SIMPLE_NETWORK_PROTOCOL instance.
240   @param HeaderSize The size, in bytes, of the media header to be filled in by the
241                     Transmit() function. If HeaderSize is nonzero, then it must
242                     be equal to This->Mode->MediaHeaderSize and the DestAddr and
243                     Protocol parameters must not be NULL.
244   @param BufferSize The size, in bytes, of the entire packet (media header and
245                     data) to be transmitted through the network interface.
246   @param Buffer     A pointer to the packet (media header followed by data) to be
247                     transmitted. This parameter cannot be NULL. If HeaderSize is
248                     zero, then the media header in Buffer must already be filled
249                     in by the caller. If HeaderSize is nonzero, then the media
250                     header will be filled in by the Transmit() function.
251   @param SrcAddr    The source HW MAC address. If HeaderSize is zero, then this
252                     parameter is ignored. If HeaderSize is nonzero and SrcAddr
253                     is NULL, then This->Mode->CurrentAddress is used for the
254                     source HW MAC address.
255   @param DestAddr   The destination HW MAC address. If HeaderSize is zero, then
256                     this parameter is ignored.
257   @param Protocol   The type of header to build. If HeaderSize is zero, then this
258                     parameter is ignored. See RFC 1700, section "Ether Types,"
259                     for examples.
260 
261   @retval EFI_SUCCESS           The packet was placed on the transmit queue.
262   @retval EFI_NOT_STARTED       The network interface has not been started.
263   @retval EFI_NOT_READY         The network interface is too busy to accept this
264                                 transmit request.
265   @retval EFI_BUFFER_TOO_SMALL  The BufferSize parameter is too small.
266   @retval EFI_INVALID_PARAMETER One or more of the parameters has an unsupported
267                                 value.
268   @retval EFI_DEVICE_ERROR      The command could not be sent to the network interface.
269   @retval EFI_UNSUPPORTED       This function is not supported by the network interface.
270 
271 **/
272 EFI_STATUS
273 EFIAPI
SnpUndi32Transmit(IN EFI_SIMPLE_NETWORK_PROTOCOL * This,IN UINTN HeaderSize,IN UINTN BufferSize,IN VOID * Buffer,IN EFI_MAC_ADDRESS * SrcAddr,OPTIONAL IN EFI_MAC_ADDRESS * DestAddr,OPTIONAL IN UINT16 * Protocol OPTIONAL)274 SnpUndi32Transmit (
275   IN EFI_SIMPLE_NETWORK_PROTOCOL *This,
276   IN UINTN                       HeaderSize,
277   IN UINTN                       BufferSize,
278   IN VOID                        *Buffer,
279   IN EFI_MAC_ADDRESS             *SrcAddr,  OPTIONAL
280   IN EFI_MAC_ADDRESS             *DestAddr, OPTIONAL
281   IN UINT16                      *Protocol  OPTIONAL
282   )
283 {
284   SNP_DRIVER  *Snp;
285   EFI_STATUS  Status;
286   EFI_TPL     OldTpl;
287 
288   if (This == NULL) {
289     return EFI_INVALID_PARAMETER;
290   }
291 
292   Snp = EFI_SIMPLE_NETWORK_DEV_FROM_THIS (This);
293 
294   OldTpl = gBS->RaiseTPL (TPL_CALLBACK);
295 
296   if (Snp == NULL) {
297     return EFI_DEVICE_ERROR;
298   }
299 
300   switch (Snp->Mode.State) {
301   case EfiSimpleNetworkInitialized:
302     break;
303 
304   case EfiSimpleNetworkStopped:
305     Status = EFI_NOT_STARTED;
306     goto ON_EXIT;
307 
308   default:
309     Status = EFI_DEVICE_ERROR;
310     goto ON_EXIT;
311   }
312 
313   if (Buffer == NULL) {
314     Status = EFI_INVALID_PARAMETER;
315     goto ON_EXIT;
316   }
317 
318   if (BufferSize < Snp->Mode.MediaHeaderSize) {
319     Status = EFI_BUFFER_TOO_SMALL;
320     goto ON_EXIT;
321   }
322 
323   //
324   // if the HeaderSize is non-zero, we need to fill up the header and for that
325   // we need the destination address and the protocol
326   //
327   if (HeaderSize != 0) {
328     if (HeaderSize != Snp->Mode.MediaHeaderSize || DestAddr == 0 || Protocol == 0) {
329       Status = EFI_INVALID_PARAMETER;
330       goto ON_EXIT;
331     }
332 
333     Status = PxeFillHeader (
334               Snp,
335               Buffer,
336               HeaderSize,
337               (UINT8 *) Buffer + HeaderSize,
338               BufferSize - HeaderSize,
339               DestAddr,
340               SrcAddr,
341               Protocol
342               );
343 
344     if (EFI_ERROR (Status)) {
345       goto ON_EXIT;
346     }
347   }
348 
349   Status = PxeTransmit (Snp, Buffer, BufferSize);
350 
351 ON_EXIT:
352   gBS->RestoreTPL (OldTpl);
353 
354   return Status;
355 }
356