/** @file KCS Transport Hook. Copyright (c) 2018, Intel Corporation. All rights reserved.
This program and the accompanying materials are licensed and made available under the terms and conditions of the BSD License that accompanies this distribution. The full text of the license may be found at http://opensource.org/licenses/bsd-license.php. THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED. **/ #include "KcsBmc.h" #include EFI_STATUS KcsErrorExit ( UINT64 KcsTimeoutPeriod, UINT16 KcsPort ) /*++ Routine Description: Check the KCS error status Arguments: KcsPort - The base port of KCS Returns: EFI_DEVICE_ERROR - The device error happened EFI_SUCCESS - Successfully check the KCS error status --*/ { EFI_STATUS Status; UINT8 KcsData; EFI_KCS_STATUS KcsStatus; UINT8 BmcStatus; UINT8 RetryCount; UINT64 TimeOut; TimeOut = 0; RetryCount = 0; while (RetryCount < KCS_ABORT_RETRY_COUNT) { do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsPort + 1); if (KcsStatus.RawData == 0xFF || (TimeOut >= KcsTimeoutPeriod)) { RetryCount = KCS_ABORT_RETRY_COUNT; break; } TimeOut++; } while (KcsStatus.Status.Ibf); if (RetryCount >= KCS_ABORT_RETRY_COUNT) { break; } KcsData = KCS_ABORT; IoWrite8 ((KcsPort + 1), KcsData); TimeOut = 0; do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsPort + 1); if (KcsStatus.RawData == 0xFF || (TimeOut >= KcsTimeoutPeriod)) { Status = EFI_DEVICE_ERROR; goto LabelError; } TimeOut++; } while (KcsStatus.Status.Ibf); KcsData = IoRead8 (KcsPort); KcsData = 0x0; IoWrite8 (KcsPort, KcsData); TimeOut = 0; do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsPort + 1); if (KcsStatus.RawData == 0xFF || (TimeOut >= KcsTimeoutPeriod)) { Status = EFI_DEVICE_ERROR; goto LabelError; } TimeOut++; } while (KcsStatus.Status.Ibf); if (KcsStatus.Status.State == KcsReadState) { TimeOut = 0; do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsPort + 1); if (KcsStatus.RawData == 0xFF || (TimeOut >= KcsTimeoutPeriod)) { Status = EFI_DEVICE_ERROR; goto LabelError; } TimeOut++; } while (!KcsStatus.Status.Obf); BmcStatus = IoRead8 (KcsPort); KcsData = KCS_READ; IoWrite8 (KcsPort, KcsData); TimeOut = 0; do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsPort + 1); if (KcsStatus.RawData == 0xFF || (TimeOut >= KcsTimeoutPeriod)) { Status = EFI_DEVICE_ERROR; goto LabelError; } TimeOut++; } while (KcsStatus.Status.Ibf); if (KcsStatus.Status.State == KcsIdleState) { TimeOut = 0; do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsPort + 1); if (KcsStatus.RawData == 0xFF || (TimeOut >= KcsTimeoutPeriod)) { Status = EFI_DEVICE_ERROR; goto LabelError; } TimeOut++; } while (!KcsStatus.Status.Obf); KcsData = IoRead8 (KcsPort); break; } else { RetryCount++; continue; } } else { RetryCount++; continue; } } if (RetryCount >= KCS_ABORT_RETRY_COUNT) { Status = EFI_DEVICE_ERROR; goto LabelError; } return EFI_SUCCESS; LabelError: return Status; } EFI_STATUS KcsCheckStatus ( UINT64 KcsTimeoutPeriod, UINT16 KcsPort, KCS_STATE KcsState, BOOLEAN *Idle ) /*++ Routine Description: Ckeck KCS status Arguments: KcsPort - The base port of KCS KcsState - The state of KCS to be checked Idle - If the KCS is idle Returns: EFI_SUCCESS - Checked the KCS status successfully --*/ { EFI_STATUS Status = 0; EFI_KCS_STATUS KcsStatus = { 0 }; UINT8 KcsData = 0; UINT64 TimeOut = 0; if(Idle == NULL ){ return EFI_INVALID_PARAMETER; } *Idle = FALSE; do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsPort + 1); if (KcsStatus.RawData == 0xFF || (TimeOut >= KcsTimeoutPeriod)) { Status = EFI_DEVICE_ERROR; goto LabelError; } TimeOut++; } while (KcsStatus.Status.Ibf); if (KcsState == KcsWriteState) { KcsData = IoRead8 (KcsPort); } if (KcsStatus.Status.State != KcsState) { if ((KcsStatus.Status.State == KcsIdleState) && (KcsState == KcsReadState)) { *Idle = TRUE; } else { Status = KcsErrorExit (KcsTimeoutPeriod, KcsPort); goto LabelError; } } if (KcsState == KcsReadState) { TimeOut = 0; do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsPort + 1); if (KcsStatus.RawData == 0xFF || (TimeOut >= KcsTimeoutPeriod)) { Status = EFI_DEVICE_ERROR; goto LabelError; } TimeOut++; } while (!KcsStatus.Status.Obf); } if (KcsState == KcsWriteState || Idle) { KcsData = IoRead8 (KcsPort); } return EFI_SUCCESS; LabelError: return Status; } EFI_STATUS SendDataToBmc ( UINT64 KcsTimeoutPeriod, UINT16 KcsPort, UINT8 *Data, UINT8 DataSize ) /*++ Routine Description: Send data to BMC Arguments: Data - The data pointer to be sent DataSize - The data size Returns: EFI_SUCCESS - Send out the data successfully --*/ { EFI_KCS_STATUS KcsStatus; UINT8 KcsData; UINT16 KcsIoBase; EFI_STATUS Status; UINT8 i; BOOLEAN Idle; UINT64 TimeOut = 0; DEBUG ((DEBUG_INFO, "SendDataToBmc (%ld, 0x%x) - ", KcsTimeoutPeriod, KcsPort)); for (i = 0; i < DataSize; i++) { DEBUG ((DEBUG_INFO, "%02x ", Data[i])); } DEBUG ((DEBUG_INFO, "\n")); KcsIoBase = KcsPort; do { MicroSecondDelay(KCS_DELAY_UNIT); KcsStatus.RawData = IoRead8 (KcsIoBase + 1); if ((KcsStatus.RawData == 0xFF) || (TimeOut >= KcsTimeoutPeriod)) { if ((Status = KcsErrorExit (KcsTimeoutPeriod, KcsIoBase)) != EFI_SUCCESS) { DEBUG ((DEBUG_INFO, "KcsErrorExit - %r\n", Status)); return Status; } } TimeOut++; } while (KcsStatus.Status.Ibf); KcsData = KCS_WRITE_START; IoWrite8 ((KcsIoBase + 1), KcsData); if ((Status = KcsCheckStatus (KcsTimeoutPeriod, KcsIoBase, KcsWriteState, &Idle)) != EFI_SUCCESS) { DEBUG ((DEBUG_INFO, "KcsCheckStatus 1 - %r\n", Status)); return Status; } for (i = 0; i < DataSize; i++) { if (i == (DataSize - 1)) { if ((Status = KcsCheckStatus (KcsTimeoutPeriod, KcsIoBase, KcsWriteState, &Idle)) != EFI_SUCCESS) { DEBUG ((DEBUG_INFO, "KcsCheckStatus 2 - %r\n", Status)); return Status; } KcsData = KCS_WRITE_END; IoWrite8 ((KcsIoBase + 1), KcsData); } Status = KcsCheckStatus (KcsTimeoutPeriod, KcsIoBase, KcsWriteState, &Idle); if (EFI_ERROR (Status)) { DEBUG ((DEBUG_INFO, "KcsCheckStatus 3 - %r\n", Status)); return Status; } IoWrite8 (KcsIoBase, Data[i]); } return EFI_SUCCESS; } EFI_STATUS ReceiveBmcData ( UINT64 KcsTimeoutPeriod, UINT16 KcsPort, UINT8 *Data, UINT8 *DataSize ) /*++ Routine Description: Routine Description: Receive data from BMC Arguments: Data - The buffer pointer DataSize - The buffer size Returns: EFI_SUCCESS - Received data successfully --*/ { UINT8 KcsData; UINT16 KcsIoBase; EFI_STATUS Status; BOOLEAN Idle; UINT8 Count; Count = 0; KcsIoBase = KcsPort; DEBUG ((DEBUG_INFO, "ReceiveBmcData (%ld, 0x%x)...\n", KcsTimeoutPeriod, KcsPort)); while (TRUE) { if ((Status = KcsCheckStatus (KcsTimeoutPeriod, KcsIoBase, KcsReadState, &Idle)) != EFI_SUCCESS) { DEBUG ((DEBUG_INFO, "KcsCheckStatus - %r\n", Status)); return Status; } if (Idle) { DEBUG ((DEBUG_INFO, "DataSize - 0x%x\n", Count)); *DataSize = Count; break; } if (Count > *DataSize) { DEBUG ((DEBUG_INFO, "ERROR: Count(0x%x) > *DataSize(0x%x)\n", Count, *DataSize)); return EFI_DEVICE_ERROR; } Data[Count] = IoRead8 (KcsIoBase); Count++; KcsData = KCS_READ; IoWrite8 (KcsIoBase, KcsData); } DEBUG ((DEBUG_INFO, "ReceiveBmcData (%ld, 0x%x) - ", KcsTimeoutPeriod, KcsPort)); for (Count = 0; Count < *DataSize; Count++) { DEBUG ((DEBUG_INFO, "%02x ", Data[Count])); } DEBUG ((DEBUG_INFO, "\n")); return EFI_SUCCESS; } EFI_STATUS ReceiveBmcDataFromPort ( UINT64 KcsTimeoutPeriod, UINT16 KcsPort, UINT8 *Data, UINT8 *DataSize ) /*++ Routine Description: Receive data from BMC Arguments: Data - The buffer pointer to receive data DataSize - The buffer size Returns: EFI_SUCCESS - Received the data successfully --*/ { EFI_STATUS Status; UINT16 KcsIoBase; UINT8 i; UINT8 MyDataSize; MyDataSize = *DataSize; KcsIoBase = KcsPort; for (i = 0; i < KCS_ABORT_RETRY_COUNT; i++) { Status = ReceiveBmcData (KcsTimeoutPeriod, KcsIoBase, Data, DataSize); if (EFI_ERROR (Status)) { if ((Status = KcsErrorExit (KcsTimeoutPeriod, KcsIoBase)) != EFI_SUCCESS) { return Status; } *DataSize = MyDataSize; } else { return Status; } } return EFI_DEVICE_ERROR; } EFI_STATUS SendDataToBmcPort ( UINT64 KcsTimeoutPeriod, UINT16 KcsPort, UINT8 *Data, UINT8 DataSize ) /*++ Routine Description: Send data to BMC Arguments: Data - The data pointer to be sent DataSize - The data size Returns: EFI_SUCCESS - Send out the data successfully --*/ { EFI_STATUS Status; UINT16 KcsIoBase; UINT8 i; KcsIoBase = KcsPort; for (i = 0; i < KCS_ABORT_RETRY_COUNT; i++) { Status = SendDataToBmc (KcsTimeoutPeriod, KcsIoBase, Data, DataSize); if (EFI_ERROR (Status)) { if ((Status = KcsErrorExit (KcsTimeoutPeriod, KcsIoBase)) != EFI_SUCCESS) { return Status; } } else { return Status; } } return EFI_DEVICE_ERROR; }