Blame ls-map.c

Packit 9fb349
/*
Packit 9fb349
 *	The PCI Utilities -- Bus Mapping Mode
Packit 9fb349
 *
Packit 9fb349
 *	Copyright (c) 1997--2008 Martin Mares <mj@ucw.cz>
Packit 9fb349
 *
Packit 9fb349
 *	Can be freely distributed and used under the terms of the GNU GPL.
Packit 9fb349
 */
Packit 9fb349
Packit 9fb349
#include <stdio.h>
Packit 9fb349
#include <string.h>
Packit 9fb349
#include <stdlib.h>
Packit 9fb349
Packit 9fb349
#include "lspci.h"
Packit 9fb349
Packit 9fb349
struct bus_bridge {
Packit 9fb349
  struct bus_bridge *next;
Packit 9fb349
  byte this, dev, func, first, last, bug;
Packit 9fb349
};
Packit 9fb349
Packit 9fb349
struct bus_info {
Packit 9fb349
  byte exists;
Packit 9fb349
  byte guestbook;
Packit 9fb349
  struct bus_bridge *bridges, *via;
Packit 9fb349
};
Packit 9fb349
Packit 9fb349
static struct bus_info *bus_info;
Packit 9fb349
Packit 9fb349
static void
Packit 9fb349
map_bridge(struct bus_info *bi, struct device *d, int np, int ns, int nl)
Packit 9fb349
{
Packit 9fb349
  struct bus_bridge *b = xmalloc(sizeof(struct bus_bridge));
Packit 9fb349
  struct pci_dev *p = d->dev;
Packit 9fb349
Packit 9fb349
  b->next = bi->bridges;
Packit 9fb349
  bi->bridges = b;
Packit 9fb349
  b->this = get_conf_byte(d, np);
Packit 9fb349
  b->dev = p->dev;
Packit 9fb349
  b->func = p->func;
Packit 9fb349
  b->first = get_conf_byte(d, ns);
Packit 9fb349
  b->last = get_conf_byte(d, nl);
Packit 9fb349
  printf("## %02x:%02x.%d is a bridge from %02x to %02x-%02x\n",
Packit 9fb349
	 p->bus, p->dev, p->func, b->this, b->first, b->last);
Packit 9fb349
  if (b->this != p->bus)
Packit 9fb349
    printf("!!! Bridge points to invalid primary bus.\n");
Packit 9fb349
  if (b->first > b->last)
Packit 9fb349
    {
Packit 9fb349
      printf("!!! Bridge points to invalid bus range.\n");
Packit 9fb349
      b->last = b->first;
Packit 9fb349
    }
Packit 9fb349
}
Packit 9fb349
Packit 9fb349
static void
Packit 9fb349
do_map_bus(int bus)
Packit 9fb349
{
Packit 9fb349
  int dev, func;
Packit 9fb349
  int verbose = pacc->debugging;
Packit 9fb349
  struct bus_info *bi = bus_info + bus;
Packit 9fb349
  struct device *d;
Packit 9fb349
Packit 9fb349
  if (verbose)
Packit 9fb349
    printf("Mapping bus %02x\n", bus);
Packit 9fb349
  for (dev = 0; dev < 32; dev++)
Packit 9fb349
    if (filter.slot < 0 || filter.slot == dev)
Packit 9fb349
      {
Packit 9fb349
	int func_limit = 1;
Packit 9fb349
	for (func = 0; func < func_limit; func++)
Packit 9fb349
	  if (filter.func < 0 || filter.func == func)
Packit 9fb349
	    {
Packit 9fb349
	      /* XXX: Bus mapping supports only domain 0 */
Packit 9fb349
	      struct pci_dev *p = pci_get_dev(pacc, 0, bus, dev, func);
Packit 9fb349
	      u16 vendor = pci_read_word(p, PCI_VENDOR_ID);
Packit 9fb349
	      if (vendor && vendor != 0xffff)
Packit 9fb349
		{
Packit 9fb349
		  if (!func && (pci_read_byte(p, PCI_HEADER_TYPE) & 0x80))
Packit 9fb349
		    func_limit = 8;
Packit 9fb349
		  if (verbose)
Packit 9fb349
		    printf("Discovered device %02x:%02x.%d\n", bus, dev, func);
Packit 9fb349
		  bi->exists = 1;
Packit 9fb349
		  if (d = scan_device(p))
Packit 9fb349
		    {
Packit 9fb349
		      show_device(d);
Packit 9fb349
		      switch (get_conf_byte(d, PCI_HEADER_TYPE) & 0x7f)
Packit 9fb349
			{
Packit 9fb349
			case PCI_HEADER_TYPE_BRIDGE:
Packit 9fb349
			  map_bridge(bi, d, PCI_PRIMARY_BUS, PCI_SECONDARY_BUS, PCI_SUBORDINATE_BUS);
Packit 9fb349
			  break;
Packit 9fb349
			case PCI_HEADER_TYPE_CARDBUS:
Packit 9fb349
			  map_bridge(bi, d, PCI_CB_PRIMARY_BUS, PCI_CB_CARD_BUS, PCI_CB_SUBORDINATE_BUS);
Packit 9fb349
			  break;
Packit 9fb349
			}
Packit 9fb349
		      free(d);
Packit 9fb349
		    }
Packit 9fb349
		  else if (verbose)
Packit 9fb349
		    printf("But it was filtered out.\n");
Packit 9fb349
		}
Packit 9fb349
	      pci_free_dev(p);
Packit 9fb349
	    }
Packit 9fb349
      }
Packit 9fb349
}
Packit 9fb349
Packit 9fb349
static void
Packit 9fb349
do_map_bridges(int bus, int min, int max)
Packit 9fb349
{
Packit 9fb349
  struct bus_info *bi = bus_info + bus;
Packit 9fb349
  struct bus_bridge *b;
Packit 9fb349
Packit 9fb349
  bi->guestbook = 1;
Packit 9fb349
  for (b=bi->bridges; b; b=b->next)
Packit 9fb349
    {
Packit 9fb349
      if (bus_info[b->first].guestbook)
Packit 9fb349
	b->bug = 1;
Packit 9fb349
      else if (b->first < min || b->last > max)
Packit 9fb349
	b->bug = 2;
Packit 9fb349
      else
Packit 9fb349
	{
Packit 9fb349
	  bus_info[b->first].via = b;
Packit 9fb349
	  do_map_bridges(b->first, b->first, b->last);
Packit 9fb349
	}
Packit 9fb349
    }
Packit 9fb349
}
Packit 9fb349
Packit 9fb349
static void
Packit 9fb349
map_bridges(void)
Packit 9fb349
{
Packit 9fb349
  int i;
Packit 9fb349
Packit 9fb349
  printf("\nSummary of buses:\n\n");
Packit 9fb349
  for (i=0; i<256; i++)
Packit 9fb349
    if (bus_info[i].exists && !bus_info[i].guestbook)
Packit 9fb349
      do_map_bridges(i, 0, 255);
Packit 9fb349
  for (i=0; i<256; i++)
Packit 9fb349
    {
Packit 9fb349
      struct bus_info *bi = bus_info + i;
Packit 9fb349
      struct bus_bridge *b = bi->via;
Packit 9fb349
Packit 9fb349
      if (bi->exists)
Packit 9fb349
	{
Packit 9fb349
	  printf("%02x: ", i);
Packit 9fb349
	  if (b)
Packit 9fb349
	    printf("Entered via %02x:%02x.%d\n", b->this, b->dev, b->func);
Packit 9fb349
	  else if (!i)
Packit 9fb349
	    printf("Primary host bus\n");
Packit 9fb349
	  else
Packit 9fb349
	    printf("Secondary host bus (?)\n");
Packit 9fb349
	}
Packit 9fb349
      for (b=bi->bridges; b; b=b->next)
Packit 9fb349
	{
Packit 9fb349
	  printf("\t%02x.%d Bridge to %02x-%02x", b->dev, b->func, b->first, b->last);
Packit 9fb349
	  switch (b->bug)
Packit 9fb349
	    {
Packit 9fb349
	    case 1:
Packit 9fb349
	      printf(" <overlap bug>");
Packit 9fb349
	      break;
Packit 9fb349
	    case 2:
Packit 9fb349
	      printf(" <crossing bug>");
Packit 9fb349
	      break;
Packit 9fb349
	    }
Packit 9fb349
	  putchar('\n');
Packit 9fb349
	}
Packit 9fb349
    }
Packit 9fb349
}
Packit 9fb349
Packit 9fb349
void
Packit 9fb349
map_the_bus(void)
Packit 9fb349
{
Packit 9fb349
  if (pacc->method == PCI_ACCESS_PROC_BUS_PCI ||
Packit 9fb349
      pacc->method == PCI_ACCESS_SYS_BUS_PCI ||
Packit 9fb349
      pacc->method == PCI_ACCESS_DUMP)
Packit 9fb349
    printf("WARNING: Bus mapping can be reliable only with direct hardware access enabled.\n\n");
Packit 9fb349
  bus_info = xmalloc(sizeof(struct bus_info) * 256);
Packit 9fb349
  memset(bus_info, 0, sizeof(struct bus_info) * 256);
Packit 9fb349
  if (filter.bus >= 0)
Packit 9fb349
    do_map_bus(filter.bus);
Packit 9fb349
  else
Packit 9fb349
    {
Packit 9fb349
      int bus;
Packit 9fb349
      for (bus=0; bus<256; bus++)
Packit 9fb349
	do_map_bus(bus);
Packit 9fb349
    }
Packit 9fb349
  map_bridges();
Packit 9fb349
}