162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Erratas to be applied for Andes CPU cores
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci *  Copyright (C) 2023 Renesas Electronics Corporation.
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Author: Lad Prabhakar <prabhakar.mahadev-lad.rj@bp.renesas.com>
862306a36Sopenharmony_ci */
962306a36Sopenharmony_ci
1062306a36Sopenharmony_ci#include <linux/memory.h>
1162306a36Sopenharmony_ci#include <linux/module.h>
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_ci#include <asm/alternative.h>
1462306a36Sopenharmony_ci#include <asm/cacheflush.h>
1562306a36Sopenharmony_ci#include <asm/errata_list.h>
1662306a36Sopenharmony_ci#include <asm/patch.h>
1762306a36Sopenharmony_ci#include <asm/processor.h>
1862306a36Sopenharmony_ci#include <asm/sbi.h>
1962306a36Sopenharmony_ci#include <asm/vendorid_list.h>
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_ci#define ANDESTECH_AX45MP_MARCHID	0x8000000000008a45UL
2262306a36Sopenharmony_ci#define ANDESTECH_AX45MP_MIMPID		0x500UL
2362306a36Sopenharmony_ci#define ANDESTECH_SBI_EXT_ANDES		0x0900031E
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_ci#define ANDES_SBI_EXT_IOCP_SW_WORKAROUND	1
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_cistatic long ax45mp_iocp_sw_workaround(void)
2862306a36Sopenharmony_ci{
2962306a36Sopenharmony_ci	struct sbiret ret;
3062306a36Sopenharmony_ci
3162306a36Sopenharmony_ci	/*
3262306a36Sopenharmony_ci	 * ANDES_SBI_EXT_IOCP_SW_WORKAROUND SBI EXT checks if the IOCP is missing and
3362306a36Sopenharmony_ci	 * cache is controllable only then CMO will be applied to the platform.
3462306a36Sopenharmony_ci	 */
3562306a36Sopenharmony_ci	ret = sbi_ecall(ANDESTECH_SBI_EXT_ANDES, ANDES_SBI_EXT_IOCP_SW_WORKAROUND,
3662306a36Sopenharmony_ci			0, 0, 0, 0, 0, 0);
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_ci	return ret.error ? 0 : ret.value;
3962306a36Sopenharmony_ci}
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_cistatic void errata_probe_iocp(unsigned int stage, unsigned long arch_id, unsigned long impid)
4262306a36Sopenharmony_ci{
4362306a36Sopenharmony_ci	static bool done;
4462306a36Sopenharmony_ci
4562306a36Sopenharmony_ci	if (!IS_ENABLED(CONFIG_ERRATA_ANDES_CMO))
4662306a36Sopenharmony_ci		return;
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_ci	if (done)
4962306a36Sopenharmony_ci		return;
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_ci	done = true;
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci	if (arch_id != ANDESTECH_AX45MP_MARCHID || impid != ANDESTECH_AX45MP_MIMPID)
5462306a36Sopenharmony_ci		return;
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ci	if (!ax45mp_iocp_sw_workaround())
5762306a36Sopenharmony_ci		return;
5862306a36Sopenharmony_ci
5962306a36Sopenharmony_ci	/* Set this just to make core cbo code happy */
6062306a36Sopenharmony_ci	riscv_cbom_block_size = 1;
6162306a36Sopenharmony_ci	riscv_noncoherent_supported();
6262306a36Sopenharmony_ci}
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_civoid __init_or_module andes_errata_patch_func(struct alt_entry *begin, struct alt_entry *end,
6562306a36Sopenharmony_ci					      unsigned long archid, unsigned long impid,
6662306a36Sopenharmony_ci					      unsigned int stage)
6762306a36Sopenharmony_ci{
6862306a36Sopenharmony_ci	if (stage == RISCV_ALTERNATIVES_BOOT)
6962306a36Sopenharmony_ci		errata_probe_iocp(stage, archid, impid);
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	/* we have nothing to patch here ATM so just return back */
7262306a36Sopenharmony_ci}
73