mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-18 16:14:04 +00:00
101 lines
2.3 KiB
C
101 lines
2.3 KiB
C
|
// SPDX-License-Identifier: GPL-2.0
|
||
|
/*
|
||
|
* Copyright (C) 2022 MediaTek Inc. All rights reserved.
|
||
|
*
|
||
|
* Author: Weijie Gao <weijie.gao@mediatek.com>
|
||
|
*/
|
||
|
|
||
|
#include <asm/io.h>
|
||
|
#include <asm/cm.h>
|
||
|
#include <asm/sections.h>
|
||
|
#include <asm/addrspace.h>
|
||
|
#include <asm/mipsmtregs.h>
|
||
|
#include <linux/sizes.h>
|
||
|
#include <time.h>
|
||
|
#include <cpu_func.h>
|
||
|
#include "launch.h"
|
||
|
#include "../mt7621.h"
|
||
|
|
||
|
/* Cluster Power Controller (CPC) offsets */
|
||
|
#define CPC_CL_OTHER 0x2010
|
||
|
#define CPC_CO_CMD 0x4000
|
||
|
|
||
|
/* CPC_CL_OTHER fields */
|
||
|
#define CPC_CL_OTHER_CORENUM_SHIFT 16
|
||
|
#define CPC_CL_OTHER_CORENUM GENMASK(23, 16)
|
||
|
|
||
|
/* CPC_CO_CMD */
|
||
|
#define PWR_UP 3
|
||
|
|
||
|
#define NUM_CORES 2
|
||
|
#define NUM_CPUS 4
|
||
|
#define WAIT_CPUS_TIMEOUT 4000
|
||
|
|
||
|
static void copy_launch_wait_code(void)
|
||
|
{
|
||
|
memset((void *)KSEG1, 0, SZ_4K);
|
||
|
|
||
|
memcpy((void *)KSEG1ADDR(LAUNCH_WAITCODE),
|
||
|
&launch_wait_code_start,
|
||
|
&launch_wait_code_end - &launch_wait_code_start);
|
||
|
|
||
|
invalidate_dcache_range(KSEG0, SZ_4K);
|
||
|
}
|
||
|
|
||
|
static void bootup_secondary_core(void)
|
||
|
{
|
||
|
void __iomem *cpcbase = (void __iomem *)KSEG1ADDR(MIPS_CPC_BASE);
|
||
|
int i;
|
||
|
|
||
|
for (i = 1; i < NUM_CORES; i++) {
|
||
|
writel(i << CPC_CL_OTHER_CORENUM_SHIFT, cpcbase + CPC_CL_OTHER);
|
||
|
writel(PWR_UP, cpcbase + CPC_CO_CMD);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void secondary_cpu_init(void)
|
||
|
{
|
||
|
void __iomem *sysc = (void __iomem *)KSEG1ADDR(SYSCTL_BASE);
|
||
|
u32 i, dual_core = 0, cpuready = 1, cpumask = 0x03;
|
||
|
ulong wait_tick;
|
||
|
struct cpulaunch_t *c;
|
||
|
|
||
|
/* Copy LAUNCH wait code used by other VPEs */
|
||
|
copy_launch_wait_code();
|
||
|
|
||
|
dual_core = readl(sysc + SYSCTL_CHIP_REV_ID_REG) & CPU_ID;
|
||
|
|
||
|
if (dual_core) {
|
||
|
/* Bootup secondary core for MT7621A */
|
||
|
cpumask = 0x0f;
|
||
|
|
||
|
/* Make BootROM/TPL redirect Core1's bootup flow to our entry point */
|
||
|
writel((uintptr_t)&_start, sysc + BOOT_SRAM_BASE_REG);
|
||
|
|
||
|
bootup_secondary_core();
|
||
|
}
|
||
|
|
||
|
/* Join the coherent domain */
|
||
|
join_coherent_domain(dual_core ? 2 : 1);
|
||
|
|
||
|
/* Bootup Core0/VPE1 */
|
||
|
boot_vpe1();
|
||
|
|
||
|
/* Wait for all CPU ready */
|
||
|
wait_tick = get_timer(0) + WAIT_CPUS_TIMEOUT;
|
||
|
|
||
|
while (time_before(get_timer(0), wait_tick)) {
|
||
|
/* CPU0 is obviously ready */
|
||
|
for (i = 1; i < NUM_CPUS; i++) {
|
||
|
c = (struct cpulaunch_t *)(KSEG0ADDR(CPULAUNCH) +
|
||
|
(i << LOG2CPULAUNCH));
|
||
|
|
||
|
if (c->flags & LAUNCH_FREADY)
|
||
|
cpuready |= BIT(i);
|
||
|
}
|
||
|
|
||
|
if ((cpuready & cpumask) == cpumask)
|
||
|
break;
|
||
|
}
|
||
|
}
|