MdeModulePkg: Fix service binding issue in TCP4 and Ip4 dxe.

v2: Handle error case in SockCreateChild and fix typo issue

when we destroy the socket Sock and its associated
protocol control block, we need to first close the
parent protocol, then remove the protocol from childHandle
and last to free any data structures that allocated in
CreateChild. But currently, we free the socket data
(Socket ConfigureState) before removing the protocol
form  the childhandle. So if the up layer want to send the
tcp reset packet in it's driver binding stop function, it will failed.

The IpInstance destroy state is redundant and may cause
ip transmit failed if up layer want to send ip packet.

Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Zhang Lubo <lubo.zhang@intel.com>
Cc: Wu Jiaxin <jiaxin.wu@intel.com>
Cc: Ye Ting <ting.ye@intel.com>
Cc: Fu Siyuan <siyuan.fu@intel.com>
Reviewed-by: Ye Ting <ting.ye@intel.com>
Reviewed-by: Fu Siyuan <siyuan.fu@intel.com>
Reviewed-by: Wu Jiaxin <jiaxin.wu@intel.com>
This commit is contained in:
Zhang Lubo 2017-03-16 18:03:36 +08:00 committed by Jiaxin Wu
parent 962e62bcd8
commit 4bb89650f5
7 changed files with 96 additions and 77 deletions

View File

@ -1,7 +1,7 @@
/** @file /** @file
The driver binding and service binding protocol for IP4 driver. The driver binding and service binding protocol for IP4 driver.
Copyright (c) 2005 - 2016, Intel Corporation. All rights reserved.<BR> Copyright (c) 2005 - 2017, Intel Corporation. All rights reserved.<BR>
(C) Copyright 2015 Hewlett-Packard Development Company, L.P.<BR> (C) Copyright 2015 Hewlett-Packard Development Company, L.P.<BR>
This program and the accompanying materials This program and the accompanying materials
@ -922,7 +922,6 @@ Ip4ServiceBindingDestroyChild (
IP4_PROTOCOL *IpInstance; IP4_PROTOCOL *IpInstance;
EFI_IP4_PROTOCOL *Ip4; EFI_IP4_PROTOCOL *Ip4;
EFI_TPL OldTpl; EFI_TPL OldTpl;
INTN State;
if ((This == NULL) || (ChildHandle == NULL)) { if ((This == NULL) || (ChildHandle == NULL)) {
return EFI_INVALID_PARAMETER; return EFI_INVALID_PARAMETER;
@ -960,13 +959,12 @@ Ip4ServiceBindingDestroyChild (
// when UDP driver is being stopped, it will destroy all // when UDP driver is being stopped, it will destroy all
// the IP child it opens. // the IP child it opens.
// //
if (IpInstance->State == IP4_STATE_DESTROY) { if (IpInstance->InDestroy) {
gBS->RestoreTPL (OldTpl); gBS->RestoreTPL (OldTpl);
return EFI_SUCCESS; return EFI_SUCCESS;
} }
State = IpInstance->State; IpInstance->InDestroy = TRUE;
IpInstance->State = IP4_STATE_DESTROY;
// //
// Close the Managed Network protocol. // Close the Managed Network protocol.
@ -1009,6 +1007,7 @@ Ip4ServiceBindingDestroyChild (
); );
OldTpl = gBS->RaiseTPL (TPL_CALLBACK); OldTpl = gBS->RaiseTPL (TPL_CALLBACK);
if (EFI_ERROR (Status)) { if (EFI_ERROR (Status)) {
IpInstance->InDestroy = FALSE;
goto ON_ERROR; goto ON_ERROR;
} }
@ -1033,7 +1032,6 @@ Ip4ServiceBindingDestroyChild (
return EFI_SUCCESS; return EFI_SUCCESS;
ON_ERROR: ON_ERROR:
IpInstance->State = State;
gBS->RestoreTPL (OldTpl); gBS->RestoreTPL (OldTpl);
return Status; return Status;

View File

@ -550,6 +550,7 @@ Ip4InitProtocol (
IpInstance->Signature = IP4_PROTOCOL_SIGNATURE; IpInstance->Signature = IP4_PROTOCOL_SIGNATURE;
CopyMem (&IpInstance->Ip4Proto, &mEfiIp4ProtocolTemplete, sizeof (IpInstance->Ip4Proto)); CopyMem (&IpInstance->Ip4Proto, &mEfiIp4ProtocolTemplete, sizeof (IpInstance->Ip4Proto));
IpInstance->State = IP4_STATE_UNCONFIGED; IpInstance->State = IP4_STATE_UNCONFIGED;
IpInstance->InDestroy = FALSE;
IpInstance->Service = IpSb; IpInstance->Service = IpSb;
InitializeListHead (&IpInstance->Link); InitializeListHead (&IpInstance->Link);
@ -936,8 +937,7 @@ EfiIp4Configure (
Status = Ip4CleanProtocol (IpInstance); Status = Ip4CleanProtocol (IpInstance);
// //
// Don't change the state if it is DESTROY, consider the following // Consider the following valid sequence: Mnp is unloaded-->Ip Stopped-->Udp Stopped,
// valid sequence: Mnp is unloaded-->Ip Stopped-->Udp Stopped,
// Configure (ThisIp, NULL). If the state is changed to UNCONFIGED, // Configure (ThisIp, NULL). If the state is changed to UNCONFIGED,
// the unload fails miserably. // the unload fails miserably.
// //

View File

@ -1,7 +1,7 @@
/** @file /** @file
Ip4 internal functions and type defintions. Ip4 internal functions and type defintions.
Copyright (c) 2005 - 2016, Intel Corporation. All rights reserved.<BR> Copyright (c) 2005 - 2017, Intel Corporation. All rights reserved.<BR>
(C) Copyright 2015 Hewlett-Packard Development Company, L.P.<BR> (C) Copyright 2015 Hewlett-Packard Development Company, L.P.<BR>
This program and the accompanying materials This program and the accompanying materials
@ -64,12 +64,10 @@ WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
// //
// The state of IP4 protocol. It starts from UNCONFIGED. if it is // The state of IP4 protocol. It starts from UNCONFIGED. if it is
// successfully configured, it goes to CONFIGED. if configure NULL // successfully configured, it goes to CONFIGED. if configure NULL
// is called, it becomes UNCONFIGED again. If (partly) destroyed, it // is called, it becomes UNCONFIGED again.
// becomes DESTROY.
// //
#define IP4_STATE_UNCONFIGED 0 #define IP4_STATE_UNCONFIGED 0
#define IP4_STATE_CONFIGED 1 #define IP4_STATE_CONFIGED 1
#define IP4_STATE_DESTROY 2
// //
// The state of IP4 service. It starts from UNSTARTED. It transits // The state of IP4 service. It starts from UNSTARTED. It transits
@ -136,6 +134,8 @@ struct _IP4_PROTOCOL {
EFI_HANDLE Handle; EFI_HANDLE Handle;
INTN State; INTN State;
BOOLEAN InDestroy;
IP4_SERVICE *Service; IP4_SERVICE *Service;
LIST_ENTRY Link; // Link to all the IP protocol from the service LIST_ENTRY Link; // Link to all the IP protocol from the service

View File

@ -1,7 +1,7 @@
/** @file /** @file
Implementation of the Socket. Implementation of the Socket.
Copyright (c) 2005 - 2016, Intel Corporation. All rights reserved.<BR> Copyright (c) 2005 - 2017, Intel Corporation. All rights reserved.<BR>
This program and the accompanying materials This program and the accompanying materials
are licensed and made available under the terms and conditions of the BSD License are licensed and made available under the terms and conditions of the BSD License
which accompanies this distribution. The full text of the license may be found at which accompanies this distribution. The full text of the license may be found at
@ -717,16 +717,8 @@ SockDestroy (
IN OUT SOCKET *Sock IN OUT SOCKET *Sock
) )
{ {
VOID *SockProtocol;
EFI_GUID *ProtocolGuid;
EFI_STATUS Status;
ASSERT (SockStream == Sock->Type); ASSERT (SockStream == Sock->Type);
if (Sock->DestroyCallback != NULL) {
Sock->DestroyCallback (Sock, Sock->Context);
}
// //
// Flush the completion token buffered // Flush the completion token buffered
// by sock and rcv, snd buffer // by sock and rcv, snd buffer
@ -762,44 +754,6 @@ SockDestroy (
Sock->Parent = NULL; Sock->Parent = NULL;
} }
//
// Set the protocol guid and driver binding handle
// in the light of Sock->SockType
//
ProtocolGuid = &gEfiTcp4ProtocolGuid;
//
// Retrieve the protocol installed on this sock
//
Status = gBS->OpenProtocol (
Sock->SockHandle,
ProtocolGuid,
&SockProtocol,
Sock->DriverBinding,
Sock->SockHandle,
EFI_OPEN_PROTOCOL_GET_PROTOCOL
);
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_ERROR, "SockDestroy: Open protocol installed "
"on socket failed with %r\n", Status));
goto FreeSock;
}
//
// Uninstall the protocol installed on this sock
// in the light of Sock->SockType
//
gBS->UninstallMultipleProtocolInterfaces (
Sock->SockHandle,
ProtocolGuid,
SockProtocol,
NULL
);
FreeSock:
FreePool (Sock); FreePool (Sock);
return ; return ;
} }

View File

@ -1,7 +1,7 @@
/** @file /** @file
Socket implementation header file. Socket implementation header file.
Copyright (c) 2005 - 2006, Intel Corporation. All rights reserved.<BR> Copyright (c) 2005 - 2017, Intel Corporation. All rights reserved.<BR>
This program and the accompanying materials This program and the accompanying materials
are licensed and made available under the terms and conditions of the BSD License are licensed and made available under the terms and conditions of the BSD License
which accompanies this distribution. The full text of the license may be found at which accompanies this distribution. The full text of the license may be found at
@ -16,6 +16,7 @@ WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
#define _SOCK_IMPL_H_ #define _SOCK_IMPL_H_
#include "Socket.h" #include "Socket.h"
#include "Tcp4Main.h"
/** /**
Signal a event with the given status. Signal a event with the given status.

View File

@ -1,7 +1,7 @@
/** @file /** @file
Interface function of the Socket. Interface function of the Socket.
Copyright (c) 2005 - 2016, Intel Corporation. All rights reserved.<BR> Copyright (c) 2005 - 2017, Intel Corporation. All rights reserved.<BR>
This program and the accompanying materials This program and the accompanying materials
are licensed and made available under the terms and conditions of the BSD License are licensed and made available under the terms and conditions of the BSD License
which accompanies this distribution. The full text of the license may be found at which accompanies this distribution. The full text of the license may be found at
@ -144,7 +144,10 @@ SockDestroyChild (
IN SOCKET *Sock IN SOCKET *Sock
) )
{ {
EFI_STATUS Status; EFI_STATUS Status;
TCP4_PROTO_DATA *ProtoData;
TCP_CB *Tcb;
VOID *SockProtocol;
ASSERT ((Sock != NULL) && (Sock->ProtoHandler != NULL)); ASSERT ((Sock != NULL) && (Sock->ProtoHandler != NULL));
@ -154,6 +157,11 @@ SockDestroyChild (
Sock->InDestroy = TRUE; Sock->InDestroy = TRUE;
ProtoData = (TCP4_PROTO_DATA *) Sock->ProtoReserved;
Tcb = ProtoData->TcpPcb;
ASSERT (Tcb != NULL);
Status = EfiAcquireLockOrFail (&(Sock->Lock)); Status = EfiAcquireLockOrFail (&(Sock->Lock));
if (EFI_ERROR (Status)) { if (EFI_ERROR (Status)) {
@ -163,6 +171,49 @@ SockDestroyChild (
return EFI_ACCESS_DENIED; return EFI_ACCESS_DENIED;
} }
//
// Close the IP protocol.
//
gBS->CloseProtocol (
Tcb->IpInfo->ChildHandle,
&gEfiIp4ProtocolGuid,
ProtoData->TcpService->IpIo->Image,
Sock->SockHandle
);
if (Sock->DestroyCallback != NULL) {
Sock->DestroyCallback (Sock, Sock->Context);
}
//
// Retrieve the protocol installed on this sock
//
Status = gBS->OpenProtocol (
Sock->SockHandle,
&gEfiTcp4ProtocolGuid,
&SockProtocol,
Sock->DriverBinding,
Sock->SockHandle,
EFI_OPEN_PROTOCOL_GET_PROTOCOL
);
if (EFI_ERROR (Status)) {
DEBUG ((EFI_D_ERROR, "SockDestroyChild: Open protocol installed "
"on socket failed with %r\n", Status));
}
//
// Uninstall the protocol installed on this sock
// in the light of Sock->SockType
//
gBS->UninstallMultipleProtocolInterfaces (
Sock->SockHandle,
&gEfiTcp4ProtocolGuid,
SockProtocol,
NULL
);
// //
// force protocol layer to detach the PCB // force protocol layer to detach the PCB
// //
@ -209,6 +260,7 @@ SockCreateChild (
) )
{ {
SOCKET *Sock; SOCKET *Sock;
VOID *SockProtocol;
EFI_STATUS Status; EFI_STATUS Status;
// //
@ -229,8 +281,7 @@ SockCreateChild (
DEBUG ((EFI_D_ERROR, "SockCreateChild: Get the lock to " DEBUG ((EFI_D_ERROR, "SockCreateChild: Get the lock to "
"access socket failed with %r\n", Status)); "access socket failed with %r\n", Status));
SockDestroy (Sock); goto ERROR;
return NULL;
} }
// //
// inform the protocol layer to attach the socket // inform the protocol layer to attach the socket
@ -243,11 +294,36 @@ SockCreateChild (
DEBUG ((EFI_D_ERROR, "SockCreateChild: Protocol failed to" DEBUG ((EFI_D_ERROR, "SockCreateChild: Protocol failed to"
" attach a socket with %r\n", Status)); " attach a socket with %r\n", Status));
SockDestroy (Sock); goto ERROR;
Sock = NULL;
} }
return Sock; return Sock;
ERROR:
if (Sock->DestroyCallback != NULL) {
Sock->DestroyCallback (Sock, Sock->Context);
}
gBS->OpenProtocol (
Sock->SockHandle,
&gEfiTcp4ProtocolGuid,
&SockProtocol,
Sock->DriverBinding,
Sock->SockHandle,
EFI_OPEN_PROTOCOL_GET_PROTOCOL
);
//
// Uninstall the protocol installed on this sock
//
gBS->UninstallMultipleProtocolInterfaces (
Sock->SockHandle,
&gEfiTcp4ProtocolGuid,
SockProtocol,
NULL
);
SockDestroy (Sock);
return NULL;
} }

View File

@ -2,7 +2,7 @@
Tcp request dispatcher implementation. Tcp request dispatcher implementation.
(C) Copyright 2014 Hewlett-Packard Development Company, L.P.<BR> (C) Copyright 2014 Hewlett-Packard Development Company, L.P.<BR>
Copyright (c) 2005 - 2014, Intel Corporation. All rights reserved.<BR> Copyright (c) 2005 - 2017, Intel Corporation. All rights reserved.<BR>
This program and the accompanying materials This program and the accompanying materials
are licensed and made available under the terms and conditions of the BSD License are licensed and made available under the terms and conditions of the BSD License
which accompanies this distribution. The full text of the license may be found at which accompanies this distribution. The full text of the license may be found at
@ -332,16 +332,6 @@ Tcp4DetachPcb (
ASSERT (Tcb != NULL); ASSERT (Tcb != NULL);
Tcp4FlushPcb (Tcb); Tcp4FlushPcb (Tcb);
//
// Close the IP protocol.
//
gBS->CloseProtocol (
Tcb->IpInfo->ChildHandle,
&gEfiIp4ProtocolGuid,
ProtoData->TcpService->IpIo->Image,
Sk->SockHandle
);
IpIoRemoveIp (ProtoData->TcpService->IpIo, Tcb->IpInfo); IpIoRemoveIp (ProtoData->TcpService->IpIo, Tcb->IpInfo);