ArmPlatformPkg/ArmJunoDxe: Set Marvell Yukon MAC address
The patch reads a valid MAC address from the Juno IOFPGA registers and pushes it into onboard Marvell Yukon NIC. Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Daniil Egranov <daniil.egranov@linaro.org> Reviewed-by: Leif Lindholm <leif.lindholm@linaro.org>
This commit is contained in:
parent
f4d575b51b
commit
a8675a19e3
@ -15,7 +15,9 @@
|
|||||||
#include "ArmJunoDxeInternal.h"
|
#include "ArmJunoDxeInternal.h"
|
||||||
#include <ArmPlatform.h>
|
#include <ArmPlatform.h>
|
||||||
|
|
||||||
|
#include <IndustryStandard/Pci.h>
|
||||||
#include <Protocol/DevicePathFromText.h>
|
#include <Protocol/DevicePathFromText.h>
|
||||||
|
#include <Protocol/PciIo.h>
|
||||||
#include <Protocol/PciRootBridgeIo.h>
|
#include <Protocol/PciRootBridgeIo.h>
|
||||||
|
|
||||||
#include <Guid/EventGroup.h>
|
#include <Guid/EventGroup.h>
|
||||||
@ -68,6 +70,290 @@ STATIC CONST EFI_PCI_ROOT_BRIDGE_DEVICE_PATH mPciRootComplexDevicePath = {
|
|||||||
|
|
||||||
EFI_EVENT mAcpiRegistration = NULL;
|
EFI_EVENT mAcpiRegistration = NULL;
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function reads PCI ID of the controller.
|
||||||
|
|
||||||
|
@param[in] PciIo PCI IO protocol handle
|
||||||
|
@param[in] PciId Looking for specified PCI ID Vendor/Device
|
||||||
|
**/
|
||||||
|
STATIC
|
||||||
|
EFI_STATUS
|
||||||
|
ReadMarvellYoukonPciId (
|
||||||
|
IN EFI_PCI_IO_PROTOCOL *PciIo,
|
||||||
|
IN UINT32 PciId
|
||||||
|
)
|
||||||
|
{
|
||||||
|
UINT32 DevicePciId;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
|
||||||
|
Status = PciIo->Pci.Read (
|
||||||
|
PciIo,
|
||||||
|
EfiPciIoWidthUint32,
|
||||||
|
PCI_VENDOR_ID_OFFSET,
|
||||||
|
1,
|
||||||
|
&DevicePciId);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (DevicePciId != PciId) {
|
||||||
|
return EFI_NOT_FOUND;
|
||||||
|
}
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function searches for Marvell Yukon NIC on the Juno
|
||||||
|
platform and returns PCI IO protocol handle for the controller.
|
||||||
|
|
||||||
|
@param[out] PciIo PCI IO protocol handle
|
||||||
|
**/
|
||||||
|
STATIC
|
||||||
|
EFI_STATUS
|
||||||
|
GetMarvellYukonPciIoProtocol (
|
||||||
|
OUT EFI_PCI_IO_PROTOCOL **PciIo
|
||||||
|
)
|
||||||
|
{
|
||||||
|
UINTN HandleCount;
|
||||||
|
EFI_HANDLE *HandleBuffer;
|
||||||
|
UINTN HIndex;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
|
||||||
|
Status = gBS->LocateHandleBuffer (
|
||||||
|
ByProtocol,
|
||||||
|
&gEfiPciIoProtocolGuid,
|
||||||
|
NULL,
|
||||||
|
&HandleCount,
|
||||||
|
&HandleBuffer);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return (Status);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (HIndex = 0; HIndex < HandleCount; ++HIndex) {
|
||||||
|
// If PciIo opened with EFI_OPEN_PROTOCOL_GET_PROTOCOL, the CloseProtocol() is not required
|
||||||
|
Status = gBS->OpenProtocol (
|
||||||
|
HandleBuffer[HIndex],
|
||||||
|
&gEfiPciIoProtocolGuid,
|
||||||
|
(VOID **) PciIo,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
EFI_OPEN_PROTOCOL_GET_PROTOCOL);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = ReadMarvellYoukonPciId (*PciIo, JUNO_MARVELL_YUKON_ID);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
gBS->FreePool (HandleBuffer);
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function restore the original controller attributes
|
||||||
|
|
||||||
|
@param[in] PciIo PCI IO protocol handle
|
||||||
|
@param[in] PciAttr PCI controller attributes.
|
||||||
|
@param[in] AcpiResDescriptor ACPI 2.0 resource descriptors for the BAR
|
||||||
|
**/
|
||||||
|
STATIC
|
||||||
|
VOID
|
||||||
|
RestorePciDev (
|
||||||
|
IN EFI_PCI_IO_PROTOCOL *PciIo,
|
||||||
|
IN UINT64 PciAttr
|
||||||
|
)
|
||||||
|
{
|
||||||
|
PciIo->Attributes (
|
||||||
|
PciIo,
|
||||||
|
EfiPciIoAttributeOperationSet,
|
||||||
|
PciAttr,
|
||||||
|
NULL
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function returns PCI MMIO base address for a controller
|
||||||
|
|
||||||
|
@param[in] PciIo PCI IO protocol handle
|
||||||
|
@param[out] PciRegBase PCI base MMIO address
|
||||||
|
**/
|
||||||
|
STATIC
|
||||||
|
EFI_STATUS
|
||||||
|
BarIsDeviceMemory (
|
||||||
|
IN EFI_PCI_IO_PROTOCOL *PciIo,
|
||||||
|
OUT UINT32 *PciRegBase
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR *AcpiResDescriptor;
|
||||||
|
EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR *AcpiCurrentDescriptor;
|
||||||
|
|
||||||
|
// Marvell Yukon's Bar0 provides base memory address for control registers
|
||||||
|
Status = PciIo->GetBarAttributes (PciIo, PCI_BAR_IDX0, NULL, (VOID**)&AcpiResDescriptor);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
AcpiCurrentDescriptor = AcpiResDescriptor;
|
||||||
|
|
||||||
|
// Search for a memory type descriptor
|
||||||
|
while (AcpiCurrentDescriptor->Desc != ACPI_END_TAG_DESCRIPTOR) {
|
||||||
|
|
||||||
|
// Check if Bar is memory type one and fetch a base address
|
||||||
|
if (AcpiCurrentDescriptor->Desc == ACPI_ADDRESS_SPACE_DESCRIPTOR &&
|
||||||
|
AcpiCurrentDescriptor->ResType == ACPI_ADDRESS_SPACE_TYPE_MEM &&
|
||||||
|
!(AcpiCurrentDescriptor->SpecificFlag & ACPI_SPECFLAG_PREFETCHABLE)) {
|
||||||
|
*PciRegBase = AcpiCurrentDescriptor->AddrRangeMin;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
Status = EFI_UNSUPPORTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
AcpiCurrentDescriptor = (EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR *) (AcpiCurrentDescriptor + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
gBS->FreePool (AcpiResDescriptor);
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function provides PCI MMIO base address, old PCI controller attributes.
|
||||||
|
|
||||||
|
@param[in] PciIo PCI IO protocol handle
|
||||||
|
@param[out] PciRegBase PCI base MMIO address
|
||||||
|
@param[out] OldPciAttr Old PCI controller attributes.
|
||||||
|
**/
|
||||||
|
STATIC
|
||||||
|
EFI_STATUS
|
||||||
|
InitPciDev (
|
||||||
|
IN EFI_PCI_IO_PROTOCOL *PciIo,
|
||||||
|
OUT UINT32 *PciRegBase,
|
||||||
|
OUT UINT64 *OldPciAttr
|
||||||
|
)
|
||||||
|
{
|
||||||
|
UINT64 AttrSupports;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
|
||||||
|
// Get controller's current attributes
|
||||||
|
Status = PciIo->Attributes (
|
||||||
|
PciIo,
|
||||||
|
EfiPciIoAttributeOperationGet,
|
||||||
|
0,
|
||||||
|
OldPciAttr);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch supported attributes
|
||||||
|
Status = PciIo->Attributes (
|
||||||
|
PciIo,
|
||||||
|
EfiPciIoAttributeOperationSupported,
|
||||||
|
0,
|
||||||
|
&AttrSupports);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enable EFI_PCI_IO_ATTRIBUTE_IO, EFI_PCI_IO_ATTRIBUTE_MEMORY and
|
||||||
|
// EFI_PCI_IO_ATTRIBUTE_BUS_MASTER bits in the PCI Config Header
|
||||||
|
AttrSupports &= EFI_PCI_DEVICE_ENABLE;
|
||||||
|
Status = PciIo->Attributes (
|
||||||
|
PciIo,
|
||||||
|
EfiPciIoAttributeOperationEnable,
|
||||||
|
AttrSupports,
|
||||||
|
NULL);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = BarIsDeviceMemory (PciIo, PciRegBase);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
RestorePciDev (PciIo, *OldPciAttr);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function reads MAC address from IOFPGA and writes it to Marvell Yukon NIC
|
||||||
|
|
||||||
|
@param[in] PciRegBase PCI base MMIO address
|
||||||
|
**/
|
||||||
|
STATIC
|
||||||
|
EFI_STATUS
|
||||||
|
WriteMacAddress (
|
||||||
|
IN UINT32 PciRegBase
|
||||||
|
)
|
||||||
|
{
|
||||||
|
UINT32 MacHigh;
|
||||||
|
UINT32 MacLow;
|
||||||
|
|
||||||
|
// Read MAC address from IOFPGA
|
||||||
|
MacHigh= MmioRead32 (ARM_JUNO_SYS_PCIGBE_H);
|
||||||
|
MacLow = MmioRead32 (ARM_JUNO_SYS_PCIGBE_L);
|
||||||
|
|
||||||
|
// Set software reset control register to protect from deactivation
|
||||||
|
// the config write state
|
||||||
|
MmioWrite16 (PciRegBase + R_CONTROL_STATUS, CS_RESET_CLR);
|
||||||
|
|
||||||
|
// Convert to Marvell MAC Address register format
|
||||||
|
MacHigh = SwapBytes32 ((MacHigh & 0xFFFF) << 16 |
|
||||||
|
(MacLow & 0xFFFF0000) >> 16);
|
||||||
|
MacLow = SwapBytes32 (MacLow) >> 16;
|
||||||
|
|
||||||
|
// Set MAC Address
|
||||||
|
MmioWrite8 (PciRegBase + R_TST_CTRL_1, TST_CFG_WRITE_ENABLE);
|
||||||
|
MmioWrite32 (PciRegBase + R_MAC, MacHigh);
|
||||||
|
MmioWrite32 (PciRegBase + R_MAC_MAINT, MacHigh);
|
||||||
|
MmioWrite32 (PciRegBase + R_MAC + R_MAC_LOW, MacLow);
|
||||||
|
MmioWrite32 (PciRegBase + R_MAC_MAINT + R_MAC_LOW, MacLow);
|
||||||
|
MmioWrite8 (PciRegBase + R_TST_CTRL_1, TST_CFG_WRITE_DISABLE);
|
||||||
|
|
||||||
|
// Initiate device reset
|
||||||
|
MmioWrite16 (PciRegBase + R_CONTROL_STATUS, CS_RESET_SET);
|
||||||
|
MmioWrite16 (PciRegBase + R_CONTROL_STATUS, CS_RESET_CLR);
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
The function reads MAC address from Juno IOFPGA registers and writes it
|
||||||
|
into Marvell Yukon NIC.
|
||||||
|
**/
|
||||||
|
STATIC
|
||||||
|
EFI_STATUS
|
||||||
|
ArmJunoSetNicMacAddress ()
|
||||||
|
{
|
||||||
|
UINT64 OldPciAttr;
|
||||||
|
EFI_PCI_IO_PROTOCOL* PciIo;
|
||||||
|
UINT32 PciRegBase;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
|
||||||
|
Status = GetMarvellYukonPciIoProtocol (&PciIo);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = InitPciDev (PciIo, &PciRegBase, &OldPciAttr);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = WriteMacAddress (PciRegBase);
|
||||||
|
|
||||||
|
RestorePciDev (PciIo, OldPciAttr);
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Notification function of the event defined as belonging to the
|
Notification function of the event defined as belonging to the
|
||||||
EFI_END_OF_DXE_EVENT_GROUP_GUID event group that was created in
|
EFI_END_OF_DXE_EVENT_GROUP_GUID event group that was created in
|
||||||
@ -106,6 +392,9 @@ OnEndOfDxe (
|
|||||||
|
|
||||||
Status = gBS->ConnectController (Handle, NULL, PciRootComplexDevicePath, FALSE);
|
Status = gBS->ConnectController (Handle, NULL, PciRootComplexDevicePath, FALSE);
|
||||||
ASSERT_EFI_ERROR (Status);
|
ASSERT_EFI_ERROR (Status);
|
||||||
|
|
||||||
|
Status = ArmJunoSetNicMacAddress ();
|
||||||
|
ASSERT_EFI_ERROR (Status);
|
||||||
}
|
}
|
||||||
|
|
||||||
STATIC
|
STATIC
|
||||||
|
@ -29,6 +29,19 @@
|
|||||||
|
|
||||||
#include <IndustryStandard/Acpi.h>
|
#include <IndustryStandard/Acpi.h>
|
||||||
|
|
||||||
|
#define ACPI_SPECFLAG_PREFETCHABLE 0x06
|
||||||
|
#define JUNO_MARVELL_YUKON_ID 0x438011AB /* Juno Marvell PCI Dev ID */
|
||||||
|
#define TST_CFG_WRITE_ENABLE 0x02 /* Enable Config Write */
|
||||||
|
#define TST_CFG_WRITE_DISABLE 0x00 /* Disable Config Write */
|
||||||
|
#define CS_RESET_CLR 0x02 /* SW Reset Clear */
|
||||||
|
#define CS_RESET_SET 0x00 /* SW Reset Set */
|
||||||
|
#define R_CONTROL_STATUS 0x0004 /* Control/Status Register */
|
||||||
|
#define R_MAC 0x0100 /* MAC Address */
|
||||||
|
#define R_MAC_MAINT 0x0110 /* MAC Address Maintenance */
|
||||||
|
#define R_MAC_LOW 0x04 /* MAC Address Low Register Offset */
|
||||||
|
#define R_TST_CTRL_1 0x0158 /* Test Control Register 1 */
|
||||||
|
|
||||||
|
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
PciEmulationEntryPoint (
|
PciEmulationEntryPoint (
|
||||||
VOID
|
VOID
|
||||||
|
Loading…
x
Reference in New Issue
Block a user