/* * This file is part of the coreboot project. * * Copyright (C) 2012 - 2017 Advanced Micro Devices, Inc. * Copyright (C) 2018 Kyösti Mälkki * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; version 2 of the License. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include void __weak SetMemParams(AMD_POST_PARAMS *PostParams) {} void __weak OemPostParams(AMD_POST_PARAMS *PostParams) {} /* ACPI table pointers returned by AmdInitLate */ static void *DmiTable; static void *AcpiPstate; static void *AcpiSrat; static void *AcpiSlit; static void *AcpiWheaMce; static void *AcpiWheaCmc; static void *AcpiAlib; static void *AcpiIvrs; static void *AcpiCrat; static AGESA_STATUS agesawrapper_readeventlog(UINT8 HeapStatus) { AGESA_STATUS Status; EVENT_PARAMS AmdEventParams = { .StdHeader.CalloutPtr = &GetBiosCallout, .StdHeader.HeapStatus = HeapStatus, }; Status = AmdReadEventLog(&AmdEventParams); if (AmdEventParams.EventClass) printk(BIOS_DEBUG, "AGESA Event Log:\n"); while (AmdEventParams.EventClass != 0) { printk(BIOS_DEBUG, " Class = %x, Info = %x," " Param1 = 0x%x, Param2 = 0x%x" " Param3 = 0x%x, Param4 = 0x%x\n", (u32)AmdEventParams.EventClass, (u32)AmdEventParams.EventInfo, (u32)AmdEventParams.DataParam1, (u32)AmdEventParams.DataParam2, (u32)AmdEventParams.DataParam3, (u32)AmdEventParams.DataParam4); Status = AmdReadEventLog(&AmdEventParams); } return Status; } static void *create_struct(AMD_INTERFACE_PARAMS *interface_struct) { /* Should clone entire StdHeader here. */ interface_struct->StdHeader.CalloutPtr = &GetBiosCallout; AGESA_STATUS status = AmdCreateStruct(interface_struct); if (status != AGESA_SUCCESS) { printk(BIOS_ERR, "Error: AmdCreateStruct() for 0x%x returned 0x%x. " "Proper system initialization may not be possible.\n", interface_struct->AgesaFunctionName, status); } if (!interface_struct->NewStructPtr) /* Avoid NULL pointer usage */ die("No AGESA structure created"); return interface_struct->NewStructPtr; } AGESA_STATUS agesawrapper_amdinitreset(void) { AGESA_STATUS status; AMD_RESET_PARAMS _ResetParams; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_INIT_RESET, .AllocationMethod = ByHost, .NewStructSize = sizeof(AMD_RESET_PARAMS), .NewStructPtr = &_ResetParams, }; AMD_RESET_PARAMS *ResetParams = create_struct(&AmdParamStruct); SetFchResetParams(&ResetParams->FchInterface); timestamp_add_now(TS_AGESA_INIT_RESET_START); status = AmdInitReset(ResetParams); timestamp_add_now(TS_AGESA_INIT_RESET_DONE); if (status != AGESA_SUCCESS) agesawrapper_readeventlog(AmdParamStruct.StdHeader.HeapStatus); AmdReleaseStruct(&AmdParamStruct); return status; } AGESA_STATUS agesawrapper_amdinitearly(void) { AGESA_STATUS status; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_INIT_EARLY, .AllocationMethod = PreMemHeap, }; AMD_EARLY_PARAMS *EarlyParams = create_struct(&AmdParamStruct); OemCustomizeInitEarly(EarlyParams); timestamp_add_now(TS_AGESA_INIT_EARLY_START); status = AmdInitEarly(EarlyParams); timestamp_add_now(TS_AGESA_INIT_EARLY_DONE); if (status != AGESA_SUCCESS) agesawrapper_readeventlog(AmdParamStruct.StdHeader.HeapStatus); AmdReleaseStruct(&AmdParamStruct); return status; } static void print_init_post_settings(AMD_POST_PARAMS *parms) { u64 syslimit, bottomio, uma_size, uma_start; const char *mode; switch (parms->MemConfig.UmaMode) { case UMA_AUTO: mode = "UMA_AUTO"; break; case UMA_SPECIFIED: mode = "UMA_SPECIFIED"; break; case UMA_NONE: mode = "UMA_NONE"; break; default: mode = "unknown!"; break; } syslimit = (u64)parms->MemConfig.SysLimit * 64 * KiB; bottomio = (u64)parms->MemConfig.BottomIo * 64 * KiB; uma_size = (u64)parms->MemConfig.UmaSize * 64 * KiB; uma_start = (u64)parms->MemConfig.UmaBase * 64 * KiB; printk(BIOS_SPEW, "AGESA set: umamode %s\n", mode); printk(BIOS_SPEW, " : syslimit 0x%llx, bottomio 0x%08llx\n", syslimit, bottomio); printk(BIOS_SPEW, " : uma size %lluMB, uma start 0x%08llx\n", uma_size / MiB, uma_start); } AGESA_STATUS agesawrapper_amdinitpost(void) { AGESA_STATUS status; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_INIT_POST, .AllocationMethod = PreMemHeap, }; AMD_POST_PARAMS *PostParams = create_struct(&AmdParamStruct); PostParams->MemConfig.UmaMode = CONFIG_GFXUMA ? UMA_AUTO : UMA_NONE; PostParams->MemConfig.UmaSize = 0; PostParams->MemConfig.BottomIo = (UINT16) (CONFIG_BOTTOMIO_POSITION >> 24); SetMemParams(PostParams); OemPostParams(PostParams); printk(BIOS_SPEW, "DRAM clear on reset: %s\n", (PostParams->MemConfig.EnableMemClr == FALSE) ? "Keep" : (PostParams->MemConfig.EnableMemClr == TRUE) ? "Clear" : "unknown" ); timestamp_add_now(TS_AGESA_INIT_POST_START); status = AmdInitPost(PostParams); timestamp_add_now(TS_AGESA_INIT_POST_DONE); /* * AGESA passes back the base and size of UMA. This is the only * opportunity to get and save these settings to be used in resource * allocation. We also need to allocate the top of low memory. * If UMA is below 4GiB, UMA base is the top of low memory, otherwise * Sub4GCachetop is the top of low memory. * With UMA_NONE we see UmaBase==0. */ uintptr_t top; if (PostParams->MemConfig.UmaBase && (PostParams->MemConfig.UmaBase < ((4ull * GiB) >> 16))) top = PostParams->MemConfig.UmaBase << 16; else top = PostParams->MemConfig.Sub4GCacheTop; backup_top_of_low_cacheable(top); save_uma_size(PostParams->MemConfig.UmaSize * 64 * KiB); save_uma_base((u64)PostParams->MemConfig.UmaBase * 64 * KiB); print_init_post_settings(PostParams); if (status != AGESA_SUCCESS) agesawrapper_readeventlog(PostParams->StdHeader.HeapStatus); AmdReleaseStruct(&AmdParamStruct); return status; } AGESA_STATUS agesawrapper_amdinitenv(void) { AGESA_STATUS status; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_INIT_ENV, .AllocationMethod = PostMemDram, }; AMD_ENV_PARAMS *EnvParams = create_struct(&AmdParamStruct); SetFchEnvParams(&EnvParams->FchInterface); SetNbEnvParams(&EnvParams->GnbEnvConfiguration); timestamp_add_now(TS_AGESA_INIT_ENV_START); status = AmdInitEnv(EnvParams); timestamp_add_now(TS_AGESA_INIT_ENV_DONE); if (status != AGESA_SUCCESS) agesawrapper_readeventlog(EnvParams->StdHeader.HeapStatus); AmdReleaseStruct(&AmdParamStruct); return status; } VOID *agesawrapper_getlateinitptr(int pick) { switch (pick) { case PICK_DMI: return DmiTable; case PICK_PSTATE: return AcpiPstate; case PICK_SRAT: return AcpiSrat; case PICK_SLIT: return AcpiSlit; case PICK_WHEA_MCE: return AcpiWheaMce; case PICK_WHEA_CMC: return AcpiWheaCmc; case PICK_ALIB: return AcpiAlib; case PICK_IVRS: return AcpiIvrs; case PICK_CRAT: return AcpiCrat; default: return NULL; } } AGESA_STATUS agesawrapper_amdinitmid(void) { AGESA_STATUS status; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_INIT_MID, .AllocationMethod = PostMemDram, }; /* Enable MMIO on AMD CPU Address Map Controller */ amd_initcpuio(); AMD_MID_PARAMS *MidParams = create_struct(&AmdParamStruct); SetFchMidParams(&MidParams->FchInterface); SetNbMidParams(&MidParams->GnbMidConfiguration); timestamp_add_now(TS_AGESA_INIT_MID_START); status = AmdInitMid(MidParams); timestamp_add_now(TS_AGESA_INIT_MID_DONE); if (status != AGESA_SUCCESS) agesawrapper_readeventlog(AmdParamStruct.StdHeader.HeapStatus); AmdReleaseStruct(&AmdParamStruct); return status; } AGESA_STATUS agesawrapper_amdinitlate(void) { AGESA_STATUS Status; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_INIT_LATE, .AllocationMethod = PostMemDram, }; /* * NOTE: if not call amdcreatestruct, the initializer * (AmdInitLateInitializer) would not be called. */ AMD_LATE_PARAMS *LateParams = create_struct(&AmdParamStruct); timestamp_add_now(TS_AGESA_INIT_LATE_START); Status = AmdInitLate(LateParams); timestamp_add_now(TS_AGESA_INIT_LATE_DONE); if (Status != AGESA_SUCCESS) { agesawrapper_readeventlog(LateParams->StdHeader.HeapStatus); ASSERT(Status == AGESA_SUCCESS); } DmiTable = LateParams->DmiTable; AcpiPstate = LateParams->AcpiPState; AcpiWheaMce = LateParams->AcpiWheaMce; AcpiWheaCmc = LateParams->AcpiWheaCmc; AcpiAlib = LateParams->AcpiAlib; AcpiIvrs = LateParams->AcpiIvrs; AcpiCrat = LateParams->AcpiCrat; printk(BIOS_DEBUG, "DmiTable:%p, AcpiPstatein: %p, AcpiSrat:%p," "AcpiSlit:%p, Mce:%p, Cmc:%p," "Alib:%p, AcpiIvrs:%p in %s\n", DmiTable, AcpiPstate, AcpiSrat, AcpiSlit, AcpiWheaMce, AcpiWheaCmc, AcpiAlib, AcpiIvrs, __func__); AmdReleaseStruct(&AmdParamStruct); return Status; } AGESA_STATUS agesawrapper_amdlaterunaptask(UINT32 Func, UINTN Data, VOID *ConfigPtr) { AGESA_STATUS Status; AP_EXE_PARAMS ApExeParams; memset(&ApExeParams, 0, sizeof(ApExeParams)); ApExeParams.StdHeader.AltImageBasePtr = 0; ApExeParams.StdHeader.CalloutPtr = &GetBiosCallout; ApExeParams.StdHeader.Func = 0; ApExeParams.StdHeader.ImageBasePtr = 0; ApExeParams.FunctionNumber = Func; ApExeParams.RelatedDataBlock = ConfigPtr; Status = AmdLateRunApTask(&ApExeParams); if (Status != AGESA_SUCCESS) { /* agesawrapper_readeventlog(); */ ASSERT(Status == AGESA_SUCCESS); } return Status; } AGESA_STATUS agesawrapper_amdinitrtb(void) { AGESA_STATUS Status; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_INIT_RTB, .AllocationMethod = PostMemDram, }; AMD_RTB_PARAMS *RtbParams = create_struct(&AmdParamStruct); timestamp_add_now(TS_AGESA_INIT_RTB_START); Status = AmdInitRtb(RtbParams); timestamp_add_now(TS_AGESA_INIT_RTB_DONE); if (Status != AGESA_SUCCESS) { agesawrapper_readeventlog(AmdParamStruct.StdHeader.HeapStatus); ASSERT(Status == AGESA_SUCCESS); } if (save_s3_info(RtbParams->S3DataBlock.NvStorage, RtbParams->S3DataBlock.NvStorageSize, RtbParams->S3DataBlock.VolatileStorage, RtbParams->S3DataBlock.VolatileStorageSize)) printk(BIOS_ERR, "S3 data not saved, resuming impossible\n"); AmdReleaseStruct(&AmdParamStruct); return Status; } AGESA_STATUS agesawrapper_amdinitresume(void) { AGESA_STATUS status; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_INIT_RESUME, .AllocationMethod = PreMemHeap, }; size_t nv_size; if (!acpi_s3_resume_allowed()) return AGESA_UNSUPPORTED; AMD_RESUME_PARAMS *InitResumeParams = create_struct(&AmdParamStruct); get_s3nv_info(&InitResumeParams->S3DataBlock.NvStorage, &nv_size); InitResumeParams->S3DataBlock.NvStorageSize = nv_size; timestamp_add_now(TS_AGESA_INIT_RESUME_START); status = AmdInitResume(InitResumeParams); timestamp_add_now(TS_AGESA_INIT_RESUME_DONE); if (status != AGESA_SUCCESS) agesawrapper_readeventlog(AmdParamStruct.StdHeader.HeapStatus); AmdReleaseStruct(&AmdParamStruct); return status; } AGESA_STATUS agesawrapper_amds3laterestore(void) { AGESA_STATUS Status; AMD_S3LATE_PARAMS _S3LateParams; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_S3LATE_RESTORE, .AllocationMethod = ByHost, .NewStructSize = sizeof(AMD_S3LATE_PARAMS), .NewStructPtr = &_S3LateParams, }; size_t vol_size; if (!acpi_s3_resume_allowed()) return AGESA_UNSUPPORTED; amd_initcpuio(); AMD_S3LATE_PARAMS *S3LateParams = create_struct(&AmdParamStruct); get_s3vol_info(&S3LateParams->S3DataBlock.VolatileStorage, &vol_size); S3LateParams->S3DataBlock.VolatileStorageSize = vol_size; timestamp_add_now(TS_AGESA_S3_LATE_START); Status = AmdS3LateRestore(S3LateParams); timestamp_add_now(TS_AGESA_S3_LATE_DONE); if (Status != AGESA_SUCCESS) { agesawrapper_readeventlog(AmdParamStruct.StdHeader.HeapStatus); ASSERT(Status == AGESA_SUCCESS); } AmdReleaseStruct(&AmdParamStruct); return Status; } AGESA_STATUS agesawrapper_amds3finalrestore(void) { AGESA_STATUS Status; AMD_S3FINAL_PARAMS _S3FinalParams; AMD_INTERFACE_PARAMS AmdParamStruct = { .AgesaFunctionName = AMD_S3FINAL_RESTORE, .AllocationMethod = ByHost, .NewStructSize = sizeof(AMD_S3FINAL_PARAMS), .NewStructPtr = &_S3FinalParams, }; size_t vol_size; if (!acpi_s3_resume_allowed()) return AGESA_UNSUPPORTED; AMD_S3FINAL_PARAMS *S3FinalParams = create_struct(&AmdParamStruct); get_s3vol_info(&S3FinalParams->S3DataBlock.VolatileStorage, &vol_size); S3FinalParams->S3DataBlock.VolatileStorageSize = vol_size; timestamp_add_now(TS_AGESA_S3_FINAL_START); Status = AmdS3FinalRestore(S3FinalParams); timestamp_add_now(TS_AGESA_S3_FINAL_DONE); if (Status != AGESA_SUCCESS) { agesawrapper_readeventlog(AmdParamStruct.StdHeader.HeapStatus); ASSERT(Status == AGESA_SUCCESS); } AmdReleaseStruct(&AmdParamStruct); return Status; } static int agesa_locate_file(const char *name, struct region_device *rdev, uint32_t type) { struct cbfsf fh; if (cbfs_boot_locate(&fh, name, &type)) return -1; cbfs_file_data(rdev, &fh); return 0; } static int agesa_locate_raw_file(const char *name, struct region_device *rdev) { return agesa_locate_file(name, rdev, CBFS_TYPE_RAW); } static int agesa_locate_stage_file_early(const char *name, struct region_device *rdev) { const size_t metadata_sz = sizeof(struct cbfs_stage); if (agesa_locate_file(name, rdev, CBFS_TYPE_STAGE)) return -1; /* Peel off the cbfs stage metadata. */ return rdev_chain(rdev, rdev, metadata_sz, region_device_sz(rdev) - metadata_sz); } static int agesa_locate_stage_file_ramstage(const char *name, struct region_device *rdev) { struct prog prog = PROG_INIT(PROG_REFCODE, name); struct rmod_stage_load rmod_agesa = { .cbmem_id = CBMEM_ID_REFCODE, .prog = &prog, }; if (acpi_is_wakeup_s3() && !IS_ENABLED(CONFIG_NO_STAGE_CACHE)) { printk(BIOS_INFO, "AGESA: Loading stage from cache\n"); // There is no way to tell if this succeeded. stage_cache_load_stage(STAGE_REFCODE, &prog); } else { if (prog_locate(&prog)) return -1; if (rmodule_stage_load(&rmod_agesa) < 0) return -1; if (!IS_ENABLED(CONFIG_NO_STAGE_CACHE)) { printk(BIOS_INFO, "AGESA: Saving stage to cache\n"); stage_cache_add(STAGE_REFCODE, &prog); } } return rdev_chain(rdev, prog_rdev(&prog), 0, region_device_sz(prog_rdev(&prog))); } static int agesa_locate_stage_file(const char *name, struct region_device *rdev) { if (!ENV_RAMSTAGE || !IS_ENABLED(CONFIG_AGESA_SPLIT_MEMORY_FILES)) return agesa_locate_stage_file_early(name, rdev); return agesa_locate_stage_file_ramstage(name, rdev); } static const char *get_agesa_cbfs_name(void) { if (!IS_ENABLED(CONFIG_AGESA_SPLIT_MEMORY_FILES)) return CONFIG_AGESA_CBFS_NAME; if (!ENV_RAMSTAGE) return CONFIG_AGESA_PRE_MEMORY_CBFS_NAME; return CONFIG_AGESA_POST_MEMORY_CBFS_NAME; } const void *agesawrapper_locate_module(const CHAR8 name[8]) { const void *agesa; const AMD_IMAGE_HEADER *image; struct region_device rdev; size_t file_size; const char *fname; int ret; fname = get_agesa_cbfs_name(); if (IS_ENABLED(CONFIG_AGESA_BINARY_PI_AS_STAGE)) ret = agesa_locate_stage_file(fname, &rdev); else ret = agesa_locate_raw_file(fname, &rdev); if (ret) return NULL; file_size = region_device_sz(&rdev); /* Assume boot device is memory mapped so the mapping can leak. */ assert(IS_ENABLED(CONFIG_BOOT_DEVICE_MEMORY_MAPPED)); agesa = rdev_mmap_full(&rdev); if (!agesa) return NULL; image = amd_find_image(agesa, agesa + file_size, 4096, name); if (!image) return NULL; return (AMD_MODULE_HEADER *)image->ModuleInfoOffset; } static MODULE_ENTRY agesa_dispatcher CAR_GLOBAL; MODULE_ENTRY agesa_get_dispatcher(void) { const AMD_MODULE_HEADER *module; static const CHAR8 id[8] = AGESA_ID; MODULE_ENTRY val = car_get_var(agesa_dispatcher); if (val != NULL) return val; module = agesawrapper_locate_module(id); if (!module) return NULL; val = module->ModuleDispatcher; car_set_var(agesa_dispatcher, val); return val; }