pcidump.c 4.32 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
/* lspci.c - List PCI devices.  */
/*
 *  GRUB  --  GRand Unified Bootloader
 *  Copyright (C) 2013  Free Software Foundation, Inc.
 *
 *  GRUB is free software: you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation, either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  GRUB is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 */

#include <grub/pci.h>
#include <grub/dl.h>
#include <grub/misc.h>
#include <grub/extcmd.h>
#include <grub/env.h>
#include <grub/mm.h>
#include <grub/i18n.h>

GRUB_MOD_LICENSE ("GPLv3+");

30 31 32 33 34 35
struct iter_cxt
{
  grub_uint32_t pciid_check_mask, pciid_check_value;
  int bus, device, function;
  int check_bus, check_device, check_function;
};
36 37 38 39 40 41 42 43 44 45 46 47

static const struct grub_arg_option options[] =
  {
    {0, 'd', 0, N_("Select device by vendor and device IDs."),
     N_("[vendor]:[device]"), ARG_TYPE_STRING},
    {0, 's', 0, N_("Select device by its position on the bus."),
     N_("[bus]:[slot][.func]"), ARG_TYPE_STRING},
    {0, 0, 0, 0, 0, 0}
  };

static int
grub_pcidump_iter (grub_pci_device_t dev, grub_pci_id_t pciid,
48
		  void *data)
49
{
50
  struct iter_cxt *ctx = data;
51 52 53
  grub_pci_address_t addr;
  int i;

54
  if ((pciid & ctx->pciid_check_mask) != ctx->pciid_check_value)
55 56
    return 0;

57
  if (ctx->check_bus && grub_pci_get_bus (dev) != ctx->bus)
58 59
    return 0;

60
  if (ctx->check_device && grub_pci_get_device (dev) != ctx->device)
61 62
    return 0;

63
  if (ctx->check_function && grub_pci_get_function (dev) != ctx->function)
64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82
    return 0;

  for (i = 0; i < 256; i += 4)
    {
      addr = grub_pci_make_address (dev, i);
      grub_printf ("%08x ", grub_pci_read (addr));
      if ((i & 0xc) == 0xc)
	grub_printf ("\n");
    }

  return 0;
}

static grub_err_t
grub_cmd_pcidump (grub_extcmd_context_t ctxt,
		  int argc __attribute__ ((unused)),
		  char **argv __attribute__ ((unused)))
{
  const char *ptr;
83 84 85 86 87 88 89 90 91 92 93
  struct iter_cxt ctx =
    {
      .pciid_check_value = 0,
      .pciid_check_mask = 0,
      .check_bus = 0,
      .check_device = 0,
      .check_function = 0,
      .bus = 0,
      .function = 0,
      .device = 0
    };
94 95 96 97

  if (ctxt->state[0].set)
    {
      ptr = ctxt->state[0].arg;
98
      ctx.pciid_check_value |= (grub_strtoul (ptr, (char **) &ptr, 16) & 0xffff);
99 100 101 102 103 104
      if (grub_errno == GRUB_ERR_BAD_NUMBER)
	{
	  grub_errno = GRUB_ERR_NONE;
	  ptr = ctxt->state[0].arg;
	}
      else
105
	ctx.pciid_check_mask |= 0xffff;
106 107 108 109 110
      if (grub_errno)
	return grub_errno;
      if (*ptr != ':')
	return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("missing `%c' symbol"), ':');
      ptr++;
111
      ctx.pciid_check_value |= (grub_strtoul (ptr, (char **) &ptr, 16) & 0xffff)
112 113 114 115
	<< 16;
      if (grub_errno == GRUB_ERR_BAD_NUMBER)
	grub_errno = GRUB_ERR_NONE;
      else
116
	ctx.pciid_check_mask |= 0xffff0000;
117 118
    }

119
  ctx.pciid_check_value &= ctx.pciid_check_mask;
120 121 122 123 124 125 126

  if (ctxt->state[1].set)
    {
      const char *optr;
      
      ptr = ctxt->state[1].arg;
      optr = ptr;
127
      ctx.bus = grub_strtoul (ptr, (char **) &ptr, 16);
128 129 130 131 132 133
      if (grub_errno == GRUB_ERR_BAD_NUMBER)
	{
	  grub_errno = GRUB_ERR_NONE;
	  ptr = optr;
	}
      else
134
	ctx.check_bus = 1;
135 136 137 138 139 140
      if (grub_errno)
	return grub_errno;
      if (*ptr != ':')
	return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("missing `%c' symbol"), ':');
      ptr++;
      optr = ptr;
141
      ctx.device = grub_strtoul (ptr, (char **) &ptr, 16);
142 143 144 145 146 147
      if (grub_errno == GRUB_ERR_BAD_NUMBER)
	{
	  grub_errno = GRUB_ERR_NONE;
	  ptr = optr;
	}
      else
148
	ctx.check_device = 1;
149 150 151
      if (*ptr == '.')
	{
	  ptr++;
152
	  ctx.function = grub_strtoul (ptr, (char **) &ptr, 16);
153 154
	  if (grub_errno)
	    return grub_errno;
155
	  ctx.check_function = 1;
156 157 158
	}
    }

159
  grub_pci_iterate (grub_pcidump_iter, &ctx);
160 161 162 163 164
  return GRUB_ERR_NONE;
}

static grub_extcmd_t cmd;

165
GRUB_MOD_INIT(pcidump)
166 167 168
{
  cmd = grub_register_extcmd ("pcidump", grub_cmd_pcidump, 0,
			      N_("[-s POSITION] [-d DEVICE]"),
169
			      N_("Show raw dump of the PCI configuration space."), options);
170 171
}

172
GRUB_MOD_FINI(pcidump)
173 174 175
{
  grub_unregister_extcmd (cmd);
}