blob: 97fd0529cd3752d1337c62a26ff7df1fa958ddd6 [file] [log] [blame]
/* $NoKeywords:$ */
/**
* @file
*
* AMD Event (Error) Log APIs, and related functions.
*
* Contains code that records and returns the events and errors.
*
* @xrefitem bom "File Content Label" "Release Content"
* @e project: AGESA
* @e sub-project: CPU
* @e \$Revision: 35136 $ @e \$Date: 2010-07-16 11:29:48 +0800 (Fri, 16 Jul 2010) $
*
*/
/*
*****************************************************************************
*
* Copyright (c) 2011, Advanced Micro Devices, Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of Advanced Micro Devices, Inc. nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL ADVANCED MICRO DEVICES, INC. BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* ***************************************************************************
*
*/
/*----------------------------------------------------------------------------------------
* M O D U L E S U S E D
*----------------------------------------------------------------------------------------
*/
#include "AGESA.h"
#include "amdlib.h"
#include "heapManager.h"
#include "GeneralServices.h"
#include "Ids.h"
#include "Filecode.h"
CODE_GROUP (G1_PEICC)
RDATA_GROUP (G1_PEICC)
#define FILECODE PROC_CPU_CPUEVENTLOG_FILECODE
/*----------------------------------------------------------------------------------------
* D E F I N I T I O N S A N D M A C R O S
*----------------------------------------------------------------------------------------
*/
#define TOTAL_EVENT_LOG_BUFFERS 16
/*----------------------------------------------------------------------------------------
* T Y P E D E F S A N D S T R U C T U R E S
*----------------------------------------------------------------------------------------
*/
/**
* A wrapper for each Event Log entry.
*/
typedef struct {
UINT16 Count; ///< Entry number
AGESA_EVENT AgesaEvent; ///< The entry itself.
} AGESA_EVENT_STRUCT;
/**
* The Event Log.
*/
typedef struct {
UINT16 ReadWriteFlag; ///< Read Write flag.
UINT16 Count; ///< The total number of active entries.
UINT16 ReadRecordPtr; ///< The next entry to read.
UINT16 WriteRecordPtr; ///< The next entry to write.
AGESA_EVENT_STRUCT AgesaEventStruct[TOTAL_EVENT_LOG_BUFFERS]; ///< The entries.
} AGESA_STRUCT_BUFFER;
/*----------------------------------------------------------------------------------------
* P R O T O T Y P E S O F L O C A L F U N C T I O N S
*----------------------------------------------------------------------------------------
*/
VOID
STATIC
GetEventLogHeapPointer (
OUT AGESA_STRUCT_BUFFER **EventLog,
IN AMD_CONFIG_PARAMS *StdHeader
);
/*---------------------------------------------------------------------------------------*/
/**
* External AGESA interface to read an Event from the Event Log.
*
* This is the implementation of the external AGESA interface entry, as a thin wrapper
* around the internal log services.
*
* @param[in] Event The event class, id, and any associated data.
*
* @retval AGESA_SUCCESS Always Succeeds.
*/
AGESA_STATUS
AmdReadEventLog (
IN EVENT_PARAMS *Event
)
{
AGESA_EVENT LogEvent;
AGESA_STATUS Status;
AGESA_TESTPOINT (TpIfAmdReadEventLogEntry, &Event->StdHeader);
ASSERT (Event != NULL);
Event->StdHeader.HeapBasePtr = HeapGetBaseAddress (&Event->StdHeader);
Status = GetEventLog (&LogEvent, &Event->StdHeader);
Event->EventClass = LogEvent.EventClass;
Event->EventInfo = LogEvent.EventInfo;
Event->DataParam1 = LogEvent.DataParam1;
Event->DataParam2 = LogEvent.DataParam2;
Event->DataParam3 = LogEvent.DataParam3;
Event->DataParam4 = LogEvent.DataParam4;
AGESA_TESTPOINT (TpIfAmdReadEventLogExit, &Event->StdHeader);
return Status;
}
/*---------------------------------------------------------------------------------------*/
/**
*
* This function prepares the Event Log for use.
*
* Allocate the memory for an event log on the heap. Set the read pointer, write pointer,
* and count to reflect the log is empty.
*
* @param[in] StdHeader Our configuration, for passing to services.
*
* @retval AGESA_SUCCESS The event log is initialized.
* @retval AGESA_ERROR Allocate Heap Buffer returned an error.
*
*/
AGESA_STATUS
EventLogInitialization (
IN AMD_CONFIG_PARAMS *StdHeader
)
{
ALLOCATE_HEAP_PARAMS AllocateHeapParams;
AGESA_STRUCT_BUFFER *AgesaEventAlloc;
AGESA_STATUS Status;
AllocateHeapParams.BufferHandle = EVENT_LOG_BUFFER_HANDLE;
AllocateHeapParams.RequestedBufferSize = sizeof (AGESA_STRUCT_BUFFER);
AllocateHeapParams.Persist = HEAP_SYSTEM_MEM;
Status = HeapAllocateBuffer (&AllocateHeapParams, StdHeader);
AgesaEventAlloc = (AGESA_STRUCT_BUFFER *) AllocateHeapParams.BufferPtr;
AgesaEventAlloc->Count = 0;
AgesaEventAlloc->ReadRecordPtr = 0;
AgesaEventAlloc->WriteRecordPtr = 0;
AgesaEventAlloc->ReadWriteFlag = 1;
return Status;
}
/*---------------------------------------------------------------------------------------*/
/**
*
* This function logs AGESA events into the event log.
*
* It will put the information in a circular buffer consisting of 16 such log
* entries. If the buffer gets full, then the next event log entry will be written
* over the oldest event log entry.
*
* @param[in] EventClass The severity of the event, its associated AGESA_STATUS.
* @param[in] EventInfo Uniquely identifies the event.
* @param[in] DataParam1 Event specific additional data
* @param[in] DataParam2 Event specific additional data
* @param[in] DataParam3 Event specific additional data
* @param[in] DataParam4 Event specific additional data
* @param[in] StdHeader Header for library and services
*
*/
VOID
PutEventLog (
IN AGESA_STATUS EventClass,
IN UINT32 EventInfo,
IN UINT32 DataParam1,
IN UINT32 DataParam2,
IN UINT32 DataParam3,
IN UINT32 DataParam4,
IN AMD_CONFIG_PARAMS *StdHeader
)
{
UINT16 Index;
AGESA_STRUCT_BUFFER *AgesaEventAlloc;
IDS_HDT_CONSOLE (MAIN_FLOW, "\n * %s Event: %08x Data: %x, %x, %x, %x\n\n",
(EventClass == AGESA_FATAL) ? "FATAL" :
(EventClass == AGESA_CRITICAL) ? "CRITICAL" :
(EventClass == AGESA_ERROR) ? "ERROR" :
(EventClass == AGESA_WARNING) ? "WARNING" :
(EventClass == AGESA_ALERT) ? "ALERT" :
(EventClass == AGESA_BOUNDS_CHK) ? "BOUNDS_CHK" :
(EventClass == AGESA_UNSUPPORTED) ? "UNSUPPORTED" :
"SUCCESS", EventInfo, DataParam1, DataParam2, DataParam3, DataParam4);
AgesaEventAlloc = NULL;
GetEventLogHeapPointer (&AgesaEventAlloc, StdHeader);
ASSERT (AgesaEventAlloc != NULL);
Index = AgesaEventAlloc->WriteRecordPtr;
// Add the new event log data into a circular buffer
AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.EventClass = EventClass;
AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.EventInfo = EventInfo;
AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.DataParam1 = DataParam1;
AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.DataParam2 = DataParam2;
AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.DataParam3 = DataParam3;
AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.DataParam4 = DataParam4;
if ((AgesaEventAlloc->WriteRecordPtr == AgesaEventAlloc->ReadRecordPtr) &&
(AgesaEventAlloc->ReadWriteFlag == 0)) {
AgesaEventAlloc->WriteRecordPtr += 1;
AgesaEventAlloc->ReadRecordPtr += 1;
if (AgesaEventAlloc->WriteRecordPtr == TOTAL_EVENT_LOG_BUFFERS) {
AgesaEventAlloc->WriteRecordPtr = 0;
AgesaEventAlloc->ReadRecordPtr = 0;
}
} else {
AgesaEventAlloc->WriteRecordPtr += 1;
if (AgesaEventAlloc->WriteRecordPtr == TOTAL_EVENT_LOG_BUFFERS) {
AgesaEventAlloc->WriteRecordPtr = 0;
}
AgesaEventAlloc->ReadWriteFlag = 0;
}
AgesaEventAlloc->Count = AgesaEventAlloc->Count + 1;
if (AgesaEventAlloc->Count <= TOTAL_EVENT_LOG_BUFFERS) {
AgesaEventAlloc->AgesaEventStruct[Index].Count = Index;
}
}
/*---------------------------------------------------------------------------------------*/
/**
*
* This function gets event logs from the circular buffer.
*
* It will read the oldest entry from the circular buffer and place that information to the structure
* pointed to by the parameter. The read pointers will be incremented to remove the entry from buffer
* so that a subsequent call will return the next entry from the buffer. If the buffer is empty the
* returned log event will have EventInfo zero, which is not a valid event id.
*
* @param[out] EventRecord The next log event.
* @param[in] StdHeader Header for library and services
*
* @retval AGESA_SUCCESS Always succeeds.
*
*/
AGESA_STATUS
GetEventLog (
OUT AGESA_EVENT *EventRecord,
IN AMD_CONFIG_PARAMS *StdHeader
)
{
UINT16 Index;
AGESA_STRUCT_BUFFER *AgesaEventAlloc;
AgesaEventAlloc = NULL;
GetEventLogHeapPointer (&AgesaEventAlloc, StdHeader);
ASSERT (AgesaEventAlloc != NULL);
if ((AgesaEventAlloc->ReadRecordPtr == AgesaEventAlloc->WriteRecordPtr) &&
(AgesaEventAlloc->ReadWriteFlag == 1)) {
// EventInfo == zero, means no more data.
LibAmdMemFill (EventRecord, 0, sizeof (AGESA_EVENT), StdHeader);
} else {
Index = AgesaEventAlloc->ReadRecordPtr;
EventRecord->EventClass = AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.EventClass;
EventRecord->EventInfo = AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.EventInfo;
EventRecord->DataParam1 = AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.DataParam1;
EventRecord->DataParam2 = AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.DataParam2;
EventRecord->DataParam3 = AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.DataParam3;
EventRecord->DataParam4 = AgesaEventAlloc->AgesaEventStruct[Index].AgesaEvent.DataParam4;
if (AgesaEventAlloc->ReadRecordPtr == (TOTAL_EVENT_LOG_BUFFERS - 1)) {
AgesaEventAlloc->ReadRecordPtr = 0;
} else {
AgesaEventAlloc->ReadRecordPtr = AgesaEventAlloc->ReadRecordPtr + 1;
}
if (AgesaEventAlloc->ReadRecordPtr == AgesaEventAlloc->WriteRecordPtr) {
AgesaEventAlloc->ReadWriteFlag = 1;
}
}
return (AGESA_SUCCESS);
}
/*---------------------------------------------------------------------------------------*/
/**
*
* This function gets event logs from the circular buffer without flushing the entry.
*
* It will read the desired entry from the circular buffer and place that information to the structure
* pointed to by the parameter. The read pointers will not be incremented to remove the entry from the
* buffer. If the buffer is empty, or the desired entry does not exist, FALSE will be returned.
*
* @param[out] EventRecord The next log event.
* @param[in] Index Zero-based unread entry index
* @param[in] StdHeader Header for library and services
*
* @retval TRUE Entry exists
* @retval FALSE Entry does not exist
*
*/
BOOLEAN
PeekEventLog (
OUT AGESA_EVENT *EventRecord,
IN UINT16 Index,
IN AMD_CONFIG_PARAMS *StdHeader
)
{
UINT16 ActualIndex;
UINT16 UnreadEntries;
AGESA_STRUCT_BUFFER *AgesaEventAlloc;
AgesaEventAlloc = NULL;
GetEventLogHeapPointer (&AgesaEventAlloc, StdHeader);
ASSERT (AgesaEventAlloc != NULL);
if ((AgesaEventAlloc->ReadRecordPtr == AgesaEventAlloc->WriteRecordPtr) &&
(AgesaEventAlloc->ReadWriteFlag == 1)) {
// EventInfo == zero, means no more data.
return FALSE;
}
if (AgesaEventAlloc->ReadRecordPtr < AgesaEventAlloc->WriteRecordPtr) {
UnreadEntries = AgesaEventAlloc->WriteRecordPtr - AgesaEventAlloc->ReadRecordPtr;
} else {
UnreadEntries = TOTAL_EVENT_LOG_BUFFERS - (AgesaEventAlloc->ReadRecordPtr - AgesaEventAlloc->WriteRecordPtr);
}
if (Index >= UnreadEntries) {
return FALSE;
}
ActualIndex = Index + AgesaEventAlloc->ReadRecordPtr;
if (ActualIndex >= TOTAL_EVENT_LOG_BUFFERS) {
ActualIndex -= TOTAL_EVENT_LOG_BUFFERS;
}
EventRecord->EventClass = AgesaEventAlloc->AgesaEventStruct[ActualIndex].AgesaEvent.EventClass;
EventRecord->EventInfo = AgesaEventAlloc->AgesaEventStruct[ActualIndex].AgesaEvent.EventInfo;
EventRecord->DataParam1 = AgesaEventAlloc->AgesaEventStruct[ActualIndex].AgesaEvent.DataParam1;
EventRecord->DataParam2 = AgesaEventAlloc->AgesaEventStruct[ActualIndex].AgesaEvent.DataParam2;
EventRecord->DataParam3 = AgesaEventAlloc->AgesaEventStruct[ActualIndex].AgesaEvent.DataParam3;
EventRecord->DataParam4 = AgesaEventAlloc->AgesaEventStruct[ActualIndex].AgesaEvent.DataParam4;
return TRUE;
}
/*---------------------------------------------------------------------------------------*/
/**
*
* This function gets the Event Log pointer.
*
* It will locate the Event Log on the heap using the heap locate service. If the Event
* Log is not located, NULL is returned.
*
* @param[out] EventLog Pointer to the Event Log, or NULL.
* @param[in] StdHeader Our Configuration, for passing to services.
*
*/
VOID
STATIC
GetEventLogHeapPointer (
OUT AGESA_STRUCT_BUFFER **EventLog,
IN AMD_CONFIG_PARAMS *StdHeader
)
{
LOCATE_HEAP_PTR LocateHeapStruct;
LocateHeapStruct.BufferHandle = EVENT_LOG_BUFFER_HANDLE;
LocateHeapStruct.BufferPtr = NULL;
if ((HeapLocateBuffer (&LocateHeapStruct, StdHeader)) == AGESA_SUCCESS) {
*EventLog = (AGESA_STRUCT_BUFFER *)LocateHeapStruct.BufferPtr;
} else {
*EventLog = NULL;
}
}