hw/sparc/leon3: implement multiprocessor
This allows to register more than one CPU on the leon3_generic machine. Co-developed-by: Frederic Konrad <konrad.frederic@yahoo.fr> Signed-off-by: Clément Chigot <chigot@adacore.com> Message-ID: <20240131085047.18458-8-chigot@adacore.com> Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
This commit is contained in:
		
							parent
							
								
									92688a91c3
								
							
						
					
					
						commit
						d65aba8286
					
				@ -54,6 +54,8 @@
 | 
				
			|||||||
#define LEON3_PROM_OFFSET    (0x00000000)
 | 
					#define LEON3_PROM_OFFSET    (0x00000000)
 | 
				
			||||||
#define LEON3_RAM_OFFSET     (0x40000000)
 | 
					#define LEON3_RAM_OFFSET     (0x40000000)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define MAX_CPUS  4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define LEON3_UART_OFFSET  (0x80000100)
 | 
					#define LEON3_UART_OFFSET  (0x80000100)
 | 
				
			||||||
#define LEON3_UART_IRQ     (3)
 | 
					#define LEON3_UART_IRQ     (3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -67,7 +69,10 @@
 | 
				
			|||||||
#define LEON3_AHB_PNP_OFFSET (0xFFFFF000)
 | 
					#define LEON3_AHB_PNP_OFFSET (0xFFFFF000)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
typedef struct ResetData {
 | 
					typedef struct ResetData {
 | 
				
			||||||
 | 
					    struct CPUResetData {
 | 
				
			||||||
 | 
					        int id;
 | 
				
			||||||
        SPARCCPU *cpu;
 | 
					        SPARCCPU *cpu;
 | 
				
			||||||
 | 
					    } info[MAX_CPUS];
 | 
				
			||||||
    uint32_t entry;             /* save kernel entry in case of reset */
 | 
					    uint32_t entry;             /* save kernel entry in case of reset */
 | 
				
			||||||
} ResetData;
 | 
					} ResetData;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -123,15 +128,17 @@ static void write_bootloader(void *ptr, hwaddr kernel_addr)
 | 
				
			|||||||
    stl_p(p++, 0x01000000); /* nop */
 | 
					    stl_p(p++, 0x01000000); /* nop */
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void main_cpu_reset(void *opaque)
 | 
					static void leon3_cpu_reset(void *opaque)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    ResetData *s   = (ResetData *)opaque;
 | 
					    struct CPUResetData *info = (struct CPUResetData *) opaque;
 | 
				
			||||||
    CPUState *cpu = CPU(s->cpu);
 | 
					    int id = info->id;
 | 
				
			||||||
    CPUSPARCState  *env = &s->cpu->env;
 | 
					    ResetData *s = (ResetData *)DO_UPCAST(ResetData, info[id], info);
 | 
				
			||||||
 | 
					    CPUState *cpu = CPU(s->info[id].cpu);
 | 
				
			||||||
 | 
					    CPUSPARCState *env = cpu_env(cpu);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    cpu_reset(cpu);
 | 
					    cpu_reset(cpu);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    cpu->halted = 0;
 | 
					    cpu->halted = cpu->cpu_index != 0;
 | 
				
			||||||
    env->pc = s->entry;
 | 
					    env->pc = s->entry;
 | 
				
			||||||
    env->npc = s->entry + 4;
 | 
					    env->npc = s->entry + 4;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@ -167,8 +174,8 @@ static void leon3_cache_control_int(CPUSPARCState *env)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
static void leon3_irq_ack(CPUSPARCState *env, int intno)
 | 
					static void leon3_irq_ack(CPUSPARCState *env, int intno)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    /* No SMP support yet, only CPU #0 available so far.  */
 | 
					    CPUState *cpu = CPU(env_cpu(env));
 | 
				
			||||||
    grlib_irqmp_ack(env->irq_manager, 0, intno);
 | 
					    grlib_irqmp_ack(env->irq_manager, cpu->cpu_index, intno);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/*
 | 
					/*
 | 
				
			||||||
@ -210,6 +217,19 @@ static void leon3_set_pil_in(void *opaque, int n, int level)
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void leon3_start_cpu_async_work(CPUState *cpu, run_on_cpu_data data)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    cpu->halted = 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void leon3_start_cpu(void *opaque, int n, int level)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    CPUState *cs = CPU(opaque);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    assert(level == 1);
 | 
				
			||||||
 | 
					    async_run_on_cpu(cs, leon3_start_cpu_async_work, RUN_ON_CPU_NULL);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void leon3_irq_manager(CPUSPARCState *env, int intno)
 | 
					static void leon3_irq_manager(CPUSPARCState *env, int intno)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    leon3_irq_ack(env, intno);
 | 
					    leon3_irq_ack(env, intno);
 | 
				
			||||||
@ -235,16 +255,20 @@ static void leon3_generic_hw_init(MachineState *machine)
 | 
				
			|||||||
    AHBPnp *ahb_pnp;
 | 
					    AHBPnp *ahb_pnp;
 | 
				
			||||||
    APBPnp *apb_pnp;
 | 
					    APBPnp *apb_pnp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    reset_info = g_malloc0(sizeof(ResetData));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for (i = 0; i < machine->smp.cpus; i++) {
 | 
				
			||||||
        /* Init CPU */
 | 
					        /* Init CPU */
 | 
				
			||||||
        cpu = SPARC_CPU(cpu_create(machine->cpu_type));
 | 
					        cpu = SPARC_CPU(cpu_create(machine->cpu_type));
 | 
				
			||||||
        env = &cpu->env;
 | 
					        env = &cpu->env;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    cpu_sparc_set_id(env, 0);
 | 
					        cpu_sparc_set_id(env, i);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        /* Reset data */
 | 
					        /* Reset data */
 | 
				
			||||||
    reset_info        = g_new0(ResetData, 1);
 | 
					        reset_info->info[i].id = i;
 | 
				
			||||||
    reset_info->cpu   = cpu;
 | 
					        reset_info->info[i].cpu = cpu;
 | 
				
			||||||
    qemu_register_reset(main_cpu_reset, reset_info);
 | 
					        qemu_register_reset(leon3_cpu_reset, &reset_info->info[i]);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ahb_pnp = GRLIB_AHB_PNP(qdev_new(TYPE_GRLIB_AHB_PNP));
 | 
					    ahb_pnp = GRLIB_AHB_PNP(qdev_new(TYPE_GRLIB_AHB_PNP));
 | 
				
			||||||
    sysbus_realize_and_unref(SYS_BUS_DEVICE(ahb_pnp), &error_fatal);
 | 
					    sysbus_realize_and_unref(SYS_BUS_DEVICE(ahb_pnp), &error_fatal);
 | 
				
			||||||
@ -262,14 +286,28 @@ static void leon3_generic_hw_init(MachineState *machine)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    /* Allocate IRQ manager */
 | 
					    /* Allocate IRQ manager */
 | 
				
			||||||
    irqmpdev = qdev_new(TYPE_GRLIB_IRQMP);
 | 
					    irqmpdev = qdev_new(TYPE_GRLIB_IRQMP);
 | 
				
			||||||
 | 
					    object_property_set_int(OBJECT(irqmpdev), "ncpus", machine->smp.cpus,
 | 
				
			||||||
 | 
					                            &error_fatal);
 | 
				
			||||||
    sysbus_realize_and_unref(SYS_BUS_DEVICE(irqmpdev), &error_fatal);
 | 
					    sysbus_realize_and_unref(SYS_BUS_DEVICE(irqmpdev), &error_fatal);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for (i = 0; i < machine->smp.cpus; i++) {
 | 
				
			||||||
 | 
					        cpu = reset_info->info[i].cpu;
 | 
				
			||||||
 | 
					        env = &cpu->env;
 | 
				
			||||||
 | 
					        qdev_init_gpio_in_named_with_opaque(DEVICE(cpu), leon3_start_cpu,
 | 
				
			||||||
 | 
					                                            cpu, "start_cpu", 1);
 | 
				
			||||||
 | 
					        qdev_connect_gpio_out_named(irqmpdev, "grlib-start-cpu", i,
 | 
				
			||||||
 | 
					                                    qdev_get_gpio_in_named(DEVICE(cpu),
 | 
				
			||||||
 | 
					                                                           "start_cpu", 0));
 | 
				
			||||||
        qdev_init_gpio_in_named_with_opaque(DEVICE(cpu), leon3_set_pil_in,
 | 
					        qdev_init_gpio_in_named_with_opaque(DEVICE(cpu), leon3_set_pil_in,
 | 
				
			||||||
                                            env, "pil", 1);
 | 
					                                            env, "pil", 1);
 | 
				
			||||||
    qdev_connect_gpio_out_named(irqmpdev, "grlib-irq", 0,
 | 
					        qdev_connect_gpio_out_named(irqmpdev, "grlib-irq", i,
 | 
				
			||||||
                                qdev_get_gpio_in_named(DEVICE(cpu), "pil", 0));
 | 
					                                    qdev_get_gpio_in_named(DEVICE(cpu),
 | 
				
			||||||
    sysbus_mmio_map(SYS_BUS_DEVICE(irqmpdev), 0, LEON3_IRQMP_OFFSET);
 | 
					                                                           "pil", 0));
 | 
				
			||||||
        env->irq_manager = irqmpdev;
 | 
					        env->irq_manager = irqmpdev;
 | 
				
			||||||
        env->qemu_irq_ack = leon3_irq_manager;
 | 
					        env->qemu_irq_ack = leon3_irq_manager;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    sysbus_mmio_map(SYS_BUS_DEVICE(irqmpdev), 0, LEON3_IRQMP_OFFSET);
 | 
				
			||||||
    grlib_apb_pnp_add_entry(apb_pnp, LEON3_IRQMP_OFFSET, 0xFFF,
 | 
					    grlib_apb_pnp_add_entry(apb_pnp, LEON3_IRQMP_OFFSET, 0xFFF,
 | 
				
			||||||
                            GRLIB_VENDOR_GAISLER, GRLIB_IRQMP_DEV,
 | 
					                            GRLIB_VENDOR_GAISLER, GRLIB_IRQMP_DEV,
 | 
				
			||||||
                            2, 0, GRLIB_APBIO_AREA);
 | 
					                            2, 0, GRLIB_APBIO_AREA);
 | 
				
			||||||
@ -342,6 +380,10 @@ static void leon3_generic_hw_init(MachineState *machine)
 | 
				
			|||||||
             */
 | 
					             */
 | 
				
			||||||
            write_bootloader(memory_region_get_ram_ptr(prom), entry);
 | 
					            write_bootloader(memory_region_get_ram_ptr(prom), entry);
 | 
				
			||||||
            reset_info->entry = LEON3_PROM_OFFSET;
 | 
					            reset_info->entry = LEON3_PROM_OFFSET;
 | 
				
			||||||
 | 
					            for (i = 0; i < machine->smp.cpus; i++) {
 | 
				
			||||||
 | 
					                reset_info->info[i].cpu->env.pc = LEON3_PROM_OFFSET;
 | 
				
			||||||
 | 
					                reset_info->info[i].cpu->env.npc = LEON3_PROM_OFFSET + 4;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -380,6 +422,7 @@ static void leon3_generic_machine_init(MachineClass *mc)
 | 
				
			|||||||
    mc->init = leon3_generic_hw_init;
 | 
					    mc->init = leon3_generic_hw_init;
 | 
				
			||||||
    mc->default_cpu_type = SPARC_CPU_TYPE_NAME("LEON3");
 | 
					    mc->default_cpu_type = SPARC_CPU_TYPE_NAME("LEON3");
 | 
				
			||||||
    mc->default_ram_id = "leon3.ram";
 | 
					    mc->default_ram_id = "leon3.ram";
 | 
				
			||||||
 | 
					    mc->max_cpus = MAX_CPUS;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
DEFINE_MACHINE("leon3_generic", leon3_generic_machine_init)
 | 
					DEFINE_MACHINE("leon3_generic", leon3_generic_machine_init)
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user