Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

RAS/AMD/ATL: Expand helpers for adding and removing base and hole

The ret_addr field in struct addr_ctx contains the intermediate value of
the returned address as it passes through multiple steps in the
translation process. Currently, adding the DRAM base and legacy hole
is only done once, so it operates directly on the intermediate value.

However, for DF 4.5 non-power-of-2 denormalization, adding and removing
the DRAM base and legacy hole needs to be done for multiple temporary
address values. During this process, the intermediate value should not be
lost so the ret_addr value can't be reused.

Update the existing 'add' helper to operate on an arbitrary address
and introduce a new 'remove' helper to do the inverse operations.

Signed-off-by: John Allen <john.allen@amd.com>
Signed-off-by: Borislav Petkov (AMD) <bp@alien8.de>
Reviewed-by: Yazen Ghannam <yazen.ghannam@amd.com>
Link: https://lore.kernel.org/r/20240606203313.51197-4-john.allen@amd.com

authored by

John Allen and committed by
Borislav Petkov (AMD)
6cce048c 1233aa3f

+29 -15
+26 -15
drivers/ras/amd/atl/core.c
··· 49 49 return FIELD_GET(DF_LEGACY_MMIO_HOLE_EN, reg); 50 50 } 51 51 52 - static int add_legacy_hole(struct addr_ctx *ctx) 52 + static u64 add_legacy_hole(struct addr_ctx *ctx, u64 addr) 53 53 { 54 54 if (!legacy_hole_en(ctx)) 55 - return 0; 55 + return addr; 56 56 57 - if (ctx->ret_addr >= df_cfg.dram_hole_base) 58 - ctx->ret_addr += (BIT_ULL(32) - df_cfg.dram_hole_base); 57 + if (addr >= df_cfg.dram_hole_base) 58 + addr += (BIT_ULL(32) - df_cfg.dram_hole_base); 59 59 60 - return 0; 60 + return addr; 61 + } 62 + 63 + static u64 remove_legacy_hole(struct addr_ctx *ctx, u64 addr) 64 + { 65 + if (!legacy_hole_en(ctx)) 66 + return addr; 67 + 68 + if (addr >= df_cfg.dram_hole_base) 69 + addr -= (BIT_ULL(32) - df_cfg.dram_hole_base); 70 + 71 + return addr; 61 72 } 62 73 63 74 static u64 get_base_addr(struct addr_ctx *ctx) ··· 83 72 return base_addr << DF_DRAM_BASE_LIMIT_LSB; 84 73 } 85 74 86 - static int add_base_and_hole(struct addr_ctx *ctx) 75 + u64 add_base_and_hole(struct addr_ctx *ctx, u64 addr) 87 76 { 88 - ctx->ret_addr += get_base_addr(ctx); 77 + return add_legacy_hole(ctx, addr + get_base_addr(ctx)); 78 + } 89 79 90 - if (add_legacy_hole(ctx)) 91 - return -EINVAL; 92 - 93 - return 0; 80 + u64 remove_base_and_hole(struct addr_ctx *ctx, u64 addr) 81 + { 82 + return remove_legacy_hole(ctx, addr) - get_base_addr(ctx); 94 83 } 95 84 96 85 static bool late_hole_remove(struct addr_ctx *ctx) ··· 137 126 if (denormalize_address(&ctx)) 138 127 return -EINVAL; 139 128 140 - if (!late_hole_remove(&ctx) && add_base_and_hole(&ctx)) 141 - return -EINVAL; 129 + if (!late_hole_remove(&ctx)) 130 + ctx.ret_addr = add_base_and_hole(&ctx, ctx.ret_addr); 142 131 143 132 if (dehash_address(&ctx)) 144 133 return -EINVAL; 145 134 146 - if (late_hole_remove(&ctx) && add_base_and_hole(&ctx)) 147 - return -EINVAL; 135 + if (late_hole_remove(&ctx)) 136 + ctx.ret_addr = add_base_and_hole(&ctx, ctx.ret_addr); 148 137 149 138 if (addr_over_limit(&ctx)) 150 139 return -EINVAL;
+3
drivers/ras/amd/atl/internal.h
··· 239 239 unsigned long norm_to_sys_addr(u8 socket_id, u8 die_id, u8 coh_st_inst_id, unsigned long addr); 240 240 unsigned long convert_umc_mca_addr_to_sys_addr(struct atl_err *err); 241 241 242 + u64 add_base_and_hole(struct addr_ctx *ctx, u64 addr); 243 + u64 remove_base_and_hole(struct addr_ctx *ctx, u64 addr); 244 + 242 245 /* 243 246 * Make a gap in @data that is @num_bits long starting at @bit_num. 244 247 * e.g. data = 11111111'b