mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-04 05:50:17 +00:00 
			
		
		
		
	Signed-off-by: Wolfgang Denk <wd@denx.de> [trini: Fixup common/cmd_io.c] Signed-off-by: Tom Rini <trini@ti.com>
		
			
				
	
	
		
			265 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			265 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * (C) Copyright 2007
 | 
						|
 * Daniel Hellstrom, Gaisler Research, daniel@gaisler.com.
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier:	GPL-2.0+
 | 
						|
 */
 | 
						|
 | 
						|
/*
 | 
						|
 * AMBA Plug&Play information list command
 | 
						|
 *
 | 
						|
 */
 | 
						|
#include <common.h>
 | 
						|
#include <command.h>
 | 
						|
#include <ambapp.h>
 | 
						|
 | 
						|
DECLARE_GLOBAL_DATA_PTR;
 | 
						|
 | 
						|
/* We put these variables into .data section so that they are zero
 | 
						|
 * when entering the AMBA Plug & Play routines (in cpu/cpu/ambapp.c)
 | 
						|
 * the first time. BSS is not garantueed to be zero since BSS
 | 
						|
 * hasn't been cleared the first times entering the CPU AMBA functions.
 | 
						|
 *
 | 
						|
 * The AMBA PnP routines call these functions if ambapp_???_print is set.
 | 
						|
 *
 | 
						|
 */
 | 
						|
int ambapp_apb_print __attribute__ ((section(".data"))) = 0;
 | 
						|
int ambapp_ahb_print __attribute__ ((section(".data"))) = 0;
 | 
						|
 | 
						|
typedef struct {
 | 
						|
	int device_id;
 | 
						|
	char *name;
 | 
						|
} ambapp_device_name;
 | 
						|
 | 
						|
static ambapp_device_name gaisler_devices[] = {
 | 
						|
	{GAISLER_LEON3, "GAISLER_LEON3"},
 | 
						|
	{GAISLER_LEON3DSU, "GAISLER_LEON3DSU"},
 | 
						|
	{GAISLER_ETHAHB, "GAISLER_ETHAHB"},
 | 
						|
	{GAISLER_ETHMAC, "GAISLER_ETHMAC"},
 | 
						|
	{GAISLER_APBMST, "GAISLER_APBMST"},
 | 
						|
	{GAISLER_AHBUART, "GAISLER_AHBUART"},
 | 
						|
	{GAISLER_SRCTRL, "GAISLER_SRCTRL"},
 | 
						|
	{GAISLER_SDCTRL, "GAISLER_SDCTRL"},
 | 
						|
	{GAISLER_APBUART, "GAISLER_APBUART"},
 | 
						|
	{GAISLER_IRQMP, "GAISLER_IRQMP"},
 | 
						|
	{GAISLER_AHBRAM, "GAISLER_AHBRAM"},
 | 
						|
	{GAISLER_GPTIMER, "GAISLER_GPTIMER"},
 | 
						|
	{GAISLER_PCITRG, "GAISLER_PCITRG"},
 | 
						|
	{GAISLER_PCISBRG, "GAISLER_PCISBRG"},
 | 
						|
	{GAISLER_PCIFBRG, "GAISLER_PCIFBRG"},
 | 
						|
	{GAISLER_PCITRACE, "GAISLER_PCITRACE"},
 | 
						|
	{GAISLER_AHBTRACE, "GAISLER_AHBTRACE"},
 | 
						|
	{GAISLER_ETHDSU, "GAISLER_ETHDSU"},
 | 
						|
	{GAISLER_PIOPORT, "GAISLER_PIOPORT"},
 | 
						|
	{GAISLER_AHBJTAG, "GAISLER_AHBJTAG"},
 | 
						|
	{GAISLER_ATACTRL, "GAISLER_ATACTRL"},
 | 
						|
	{GAISLER_VGA, "GAISLER_VGA"},
 | 
						|
	{GAISLER_KBD, "GAISLER_KBD"},
 | 
						|
	{GAISLER_L2TIME, "GAISLER_L2TIME"},
 | 
						|
	{GAISLER_L2C, "GAISLER_L2C"},
 | 
						|
	{GAISLER_PLUGPLAY, "GAISLER_PLUGPLAY"},
 | 
						|
	{GAISLER_SPW, "GAISLER_SPW"},
 | 
						|
	{GAISLER_SPW2, "GAISLER_SPW2"},
 | 
						|
	{GAISLER_EHCI, "GAISLER_EHCI"},
 | 
						|
	{GAISLER_UHCI, "GAISLER_UHCI"},
 | 
						|
	{GAISLER_AHBSTAT, "GAISLER_AHBSTAT"},
 | 
						|
	{GAISLER_DDR2SPA, "GAISLER_DDR2SPA"},
 | 
						|
	{GAISLER_DDRSPA, "GAISLER_DDRSPA"},
 | 
						|
	{0, NULL}
 | 
						|
};
 | 
						|
 | 
						|
static ambapp_device_name esa_devices[] = {
 | 
						|
	{ESA_LEON2, "ESA_LEON2"},
 | 
						|
	{ESA_MCTRL, "ESA_MCTRL"},
 | 
						|
	{0, NULL}
 | 
						|
};
 | 
						|
 | 
						|
static ambapp_device_name opencores_devices[] = {
 | 
						|
	{OPENCORES_PCIBR, "OPENCORES_PCIBR"},
 | 
						|
	{OPENCORES_ETHMAC, "OPENCORES_ETHMAC"},
 | 
						|
	{0, NULL}
 | 
						|
};
 | 
						|
 | 
						|
typedef struct {
 | 
						|
	unsigned int vendor_id;
 | 
						|
	char *name;
 | 
						|
	ambapp_device_name *devices;
 | 
						|
} ambapp_vendor_devnames;
 | 
						|
 | 
						|
static ambapp_vendor_devnames vendors[] = {
 | 
						|
	{VENDOR_GAISLER, "VENDOR_GAISLER", gaisler_devices},
 | 
						|
	{VENDOR_ESA, "VENDOR_ESA", esa_devices},
 | 
						|
	{VENDOR_OPENCORES, "VENDOR_OPENCORES", opencores_devices},
 | 
						|
	{0, NULL, 0}
 | 
						|
};
 | 
						|
 | 
						|
static char *ambapp_get_devname(ambapp_device_name * devs, int id)
 | 
						|
{
 | 
						|
	if (!devs)
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	while (devs->device_id > 0) {
 | 
						|
		if (devs->device_id == id)
 | 
						|
			return devs->name;
 | 
						|
		devs++;
 | 
						|
	}
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
char *ambapp_device_id2str(int vendor, int id)
 | 
						|
{
 | 
						|
	ambapp_vendor_devnames *ven = &vendors[0];
 | 
						|
 | 
						|
	while (ven->vendor_id > 0) {
 | 
						|
		if (ven->vendor_id == vendor) {
 | 
						|
			return ambapp_get_devname(ven->devices, id);
 | 
						|
		}
 | 
						|
		ven++;
 | 
						|
	}
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
char *ambapp_vendor_id2str(int vendor)
 | 
						|
{
 | 
						|
	ambapp_vendor_devnames *ven = &vendors[0];
 | 
						|
 | 
						|
	while (ven->vendor_id > 0) {
 | 
						|
		if (ven->vendor_id == vendor) {
 | 
						|
			return ven->name;
 | 
						|
		}
 | 
						|
		ven++;
 | 
						|
	}
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
static char *unknown = "unknown";
 | 
						|
 | 
						|
/* Print one APB device */
 | 
						|
void ambapp_print_apb(apbctrl_pp_dev * apb, ambapp_ahbdev * apbmst, int index)
 | 
						|
{
 | 
						|
	char *dev_str, *ven_str;
 | 
						|
	int irq, ver, vendor, deviceid;
 | 
						|
	unsigned int address, apbmst_base, mask;
 | 
						|
 | 
						|
	vendor = amba_vendor(apb->conf);
 | 
						|
	deviceid = amba_device(apb->conf);
 | 
						|
	irq = amba_irq(apb->conf);
 | 
						|
	ver = amba_ver(apb->conf);
 | 
						|
	apbmst_base = apbmst->address[0] & LEON3_IO_AREA;
 | 
						|
	address = (apbmst_base | (((apb->bar & 0xfff00000) >> 12))) &
 | 
						|
	    (((apb->bar & 0x0000fff0) << 4) | 0xfff00000);
 | 
						|
 | 
						|
	mask = amba_membar_mask(apb->bar) << 8;
 | 
						|
	mask = ((~mask) & 0x000fffff) + 1;
 | 
						|
 | 
						|
	ven_str = ambapp_vendor_id2str(vendor);
 | 
						|
	if (!ven_str) {
 | 
						|
		ven_str = unknown;
 | 
						|
		dev_str = unknown;
 | 
						|
	} else {
 | 
						|
		dev_str = ambapp_device_id2str(vendor, deviceid);
 | 
						|
		if (!dev_str)
 | 
						|
			dev_str = unknown;
 | 
						|
	}
 | 
						|
 | 
						|
	printf("0x%02x:0x%02x:0x%02x: %s  %s\n"
 | 
						|
	       "   apb: 0x%08x - 0x%08x\n"
 | 
						|
	       "   irq: %-2d (ver: %-2d)\n",
 | 
						|
	       index, vendor, deviceid, ven_str, dev_str, address,
 | 
						|
	       address + mask, irq, ver);
 | 
						|
}
 | 
						|
 | 
						|
void ambapp_print_ahb(ahbctrl_pp_dev * ahb, int index)
 | 
						|
{
 | 
						|
	char *dev_str, *ven_str;
 | 
						|
	int irq, ver, vendor, deviceid;
 | 
						|
	unsigned int addr, mask;
 | 
						|
	int j;
 | 
						|
 | 
						|
	vendor = amba_vendor(ahb->conf);
 | 
						|
	deviceid = amba_device(ahb->conf);
 | 
						|
	irq = amba_irq(ahb->conf);
 | 
						|
	ver = amba_ver(ahb->conf);
 | 
						|
 | 
						|
	ven_str = ambapp_vendor_id2str(vendor);
 | 
						|
	if (!ven_str) {
 | 
						|
		ven_str = unknown;
 | 
						|
		dev_str = unknown;
 | 
						|
	} else {
 | 
						|
		dev_str = ambapp_device_id2str(vendor, deviceid);
 | 
						|
		if (!dev_str)
 | 
						|
			dev_str = unknown;
 | 
						|
	}
 | 
						|
 | 
						|
	printf("0x%02x:0x%02x:0x%02x: %s  %s\n",
 | 
						|
	       index, vendor, deviceid, ven_str, dev_str);
 | 
						|
 | 
						|
	for (j = 0; j < 4; j++) {
 | 
						|
		addr = amba_membar_start(ahb->bars[j]);
 | 
						|
		if (amba_membar_type(ahb->bars[j]) == 0)
 | 
						|
			continue;
 | 
						|
		if (amba_membar_type(ahb->bars[j]) == AMBA_TYPE_AHBIO)
 | 
						|
			addr = AMBA_TYPE_AHBIO_ADDR(addr);
 | 
						|
		mask = amba_membar_mask(ahb->bars[j]) << 20;
 | 
						|
		printf("   mem: 0x%08x - 0x%08x\n", addr, addr + ((~mask) + 1));
 | 
						|
	}
 | 
						|
 | 
						|
	printf("   irq: %-2d (ver: %d)\n", irq, ver);
 | 
						|
}
 | 
						|
 | 
						|
int do_ambapp_print(cmd_tbl_t * cmdtp, int flag, int argc, char * const argv[])
 | 
						|
{
 | 
						|
 | 
						|
	/* Print AHB Masters */
 | 
						|
	puts("--------- AHB Masters ---------\n");
 | 
						|
	ambapp_apb_print = 0;
 | 
						|
	ambapp_ahb_print = 1;
 | 
						|
	ambapp_ahbmst_count(99, 99);	/* Get vendor&device 99 = nonexistent... */
 | 
						|
 | 
						|
	/* Print AHB Slaves */
 | 
						|
	puts("--------- AHB Slaves  ---------\n");
 | 
						|
	ambapp_ahbslv_count(99, 99);	/* Get vendor&device 99 = nonexistent... */
 | 
						|
 | 
						|
	/* Print APB Slaves */
 | 
						|
	puts("--------- APB Slaves  ---------\n");
 | 
						|
	ambapp_apb_print = 1;
 | 
						|
	ambapp_ahb_print = 0;
 | 
						|
	ambapp_apb_count(99, 99);	/* Get vendor&device 99 = nonexistent... */
 | 
						|
 | 
						|
	/* Reset, no futher printing */
 | 
						|
	ambapp_apb_print = 0;
 | 
						|
	ambapp_ahb_print = 0;
 | 
						|
	puts("\n");
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int ambapp_init_reloc(void)
 | 
						|
{
 | 
						|
	ambapp_vendor_devnames *vend = vendors;
 | 
						|
	ambapp_device_name *dev;
 | 
						|
 | 
						|
	while (vend->vendor_id && vend->name) {
 | 
						|
		vend->name = (char *)((unsigned int)vend->name + gd->reloc_off);
 | 
						|
		vend->devices =
 | 
						|
		    (ambapp_device_name *) ((unsigned int)vend->devices +
 | 
						|
					    gd->reloc_off);;
 | 
						|
		dev = vend->devices;
 | 
						|
		vend++;
 | 
						|
		if (!dev)
 | 
						|
			continue;
 | 
						|
		while (dev->device_id && dev->name) {
 | 
						|
			dev->name =
 | 
						|
			    (char *)((unsigned int)dev->name + gd->reloc_off);;
 | 
						|
			dev++;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
U_BOOT_CMD(
 | 
						|
	ambapp, 1, 1, do_ambapp_print,
 | 
						|
	"list AMBA Plug&Play information",
 | 
						|
	"ambapp\n"
 | 
						|
	"    - lists AMBA (AHB & APB) Plug&Play devices present on the system"
 | 
						|
);
 |