blob: add1fce4c05b02adb24aa45824f76910e5f2f626 [file] [log] [blame]
Uwe Hermann515ab3d2009-05-15 17:02:34 +00001/*
2 * This file is part of the flashrom project.
3 *
4 * Copyright (C) 2009 Uwe Hermann <uwe@hermann-uwe.de>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19 */
20
21#include <stdlib.h>
22#include <string.h>
Uwe Hermann515ab3d2009-05-15 17:02:34 +000023#include <sys/types.h>
Uwe Hermann515ab3d2009-05-15 17:02:34 +000024#include "flash.h"
25
26uint32_t io_base_addr;
27struct pci_access *pacc;
28struct pci_filter filter;
Uwe Hermann8403ccb2009-05-16 21:39:19 +000029struct pci_dev *pcidev_dev = NULL;
Uwe Hermann515ab3d2009-05-15 17:02:34 +000030
Uwe Hermann2bc98f62009-09-30 18:29:55 +000031uint32_t pcidev_validate(struct pci_dev *dev, uint32_t bar,
32 struct pcidev_status *devs)
Uwe Hermann515ab3d2009-05-15 17:02:34 +000033{
34 int i;
Carl-Daniel Hailfinger295b3af2010-03-17 00:47:56 +000035 /* FIXME: 64 bit memory BARs need a 64 bit addr. */
Uwe Hermann515ab3d2009-05-15 17:02:34 +000036 uint32_t addr;
37
38 for (i = 0; devs[i].device_name != NULL; i++) {
39 if (dev->device_id != devs[i].device_id)
40 continue;
41
Uwe Hermann2bc98f62009-09-30 18:29:55 +000042 /*
43 * Don't use dev->base_addr[x] (as value for 'bar'), won't
44 * work on older libpci.
45 */
Carl-Daniel Hailfinger295b3af2010-03-17 00:47:56 +000046 addr = pci_read_long(dev, bar);
TURBO Jb0912c02009-09-02 23:00:46 +000047
Sean Nelson316a29f2010-05-07 20:09:04 +000048 msg_pinfo("Found \"%s %s\" (%04x:%04x, BDF %02x:%02x.%x).\n",
Uwe Hermann2bc98f62009-09-30 18:29:55 +000049 devs[i].vendor_name, devs[i].device_name,
50 dev->vendor_id, dev->device_id, dev->bus, dev->dev,
51 dev->func);
Carl-Daniel Hailfinger295b3af2010-03-17 00:47:56 +000052 msg_pdbg("Requested BAR is %s", (addr & 0x1) ? "IO" : "MEM");
53 if (addr & 0x1) {
54 /* Mask off IO space indicator and reserved bit. */
55 msg_pdbg("\n");
56 addr &= ~0x3;
57 } else {
58 msg_pdbg(", %sbit, %sprefetchable\n",
59 ((addr & 0x6) == 0x0) ? "32" :
60 (((addr & 0x6) == 0x4) ? "64" : "reserved"),
61 (addr & 0x8) ? "" : "not ");
62 /* Mask off Mem space indicator, 32/64bit type indicator
63 * and Prefetchable indicator.
64 */
65 addr &= ~0xf;
66 }
Uwe Hermann515ab3d2009-05-15 17:02:34 +000067
Michael Karcher84486392010-02-24 00:04:40 +000068 if (devs[i].status == NT) {
Sean Nelson316a29f2010-05-07 20:09:04 +000069 msg_pinfo("===\nThis PCI device is UNTESTED. Please "
Uwe Hermanneaefb482009-05-17 22:57:34 +000070 "report the 'flashrom -p xxxx' output \n"
Stefan Reinauer22ea8cd2009-07-30 13:32:26 +000071 "to flashrom@flashrom.org if it works "
Uwe Hermann515ab3d2009-05-15 17:02:34 +000072 "for you. Thank you for your help!\n===\n");
73 }
74
75 return addr;
76 }
77
78 return 0;
79}
80
Uwe Hermann2bc98f62009-09-30 18:29:55 +000081uint32_t pcidev_init(uint16_t vendor_id, uint32_t bar,
82 struct pcidev_status *devs, char *pcidev_bdf)
Uwe Hermann515ab3d2009-05-15 17:02:34 +000083{
84 struct pci_dev *dev;
85 char *msg = NULL;
86 int found = 0;
Uwe Hermann8403ccb2009-05-16 21:39:19 +000087 uint32_t addr = 0, curaddr = 0;
Uwe Hermann515ab3d2009-05-15 17:02:34 +000088
89 pacc = pci_alloc(); /* Get the pci_access structure */
90 pci_init(pacc); /* Initialize the PCI library */
91 pci_scan_bus(pacc); /* We want to get the list of devices */
92 pci_filter_init(pacc, &filter);
93
94 /* Filter by vendor and also bb:dd.f (if supplied by the user). */
95 filter.vendor = vendor_id;
96 if (pcidev_bdf != NULL) {
97 if ((msg = pci_filter_parse_slot(&filter, pcidev_bdf))) {
Sean Nelson316a29f2010-05-07 20:09:04 +000098 msg_perr("Error: %s\n", msg);
Uwe Hermann515ab3d2009-05-15 17:02:34 +000099 exit(1);
100 }
101 }
102
103 for (dev = pacc->devices; dev; dev = dev->next) {
104 if (pci_filter_match(&filter, dev)) {
TURBO Jb0912c02009-09-02 23:00:46 +0000105 if ((addr = pcidev_validate(dev, bar, devs)) != 0) {
Uwe Hermann8403ccb2009-05-16 21:39:19 +0000106 curaddr = addr;
107 pcidev_dev = dev;
Uwe Hermann515ab3d2009-05-15 17:02:34 +0000108 found++;
Uwe Hermann8403ccb2009-05-16 21:39:19 +0000109 }
Uwe Hermann515ab3d2009-05-15 17:02:34 +0000110 }
111 }
112
113 /* Only continue if exactly one supported PCI dev has been found. */
114 if (found == 0) {
Sean Nelson316a29f2010-05-07 20:09:04 +0000115 msg_perr("Error: No supported PCI device found.\n");
Uwe Hermann515ab3d2009-05-15 17:02:34 +0000116 exit(1);
117 } else if (found > 1) {
Sean Nelson316a29f2010-05-07 20:09:04 +0000118 msg_perr("Error: Multiple supported PCI devices found. "
Carl-Daniel Hailfinger32910302009-10-30 21:12:39 +0000119 "Use 'flashrom -p xxxx:bb:dd.f' \n"
Uwe Hermann515ab3d2009-05-15 17:02:34 +0000120 "to explicitly select the card with the given BDF "
121 "(PCI bus, device, function).\n");
122 exit(1);
123 }
124
Uwe Hermann8403ccb2009-05-16 21:39:19 +0000125 return curaddr;
Uwe Hermann515ab3d2009-05-15 17:02:34 +0000126}
127
128void print_supported_pcidevs(struct pcidev_status *devs)
129{
130 int i;
131
132 for (i = 0; devs[i].vendor_name != NULL; i++) {
Sean Nelson316a29f2010-05-07 20:09:04 +0000133 msg_pinfo("%s %s [%02x:%02x]%s\n", devs[i].vendor_name,
Uwe Hermann515ab3d2009-05-15 17:02:34 +0000134 devs[i].device_name, devs[i].vendor_id,
135 devs[i].device_id,
Michael Karcher84486392010-02-24 00:04:40 +0000136 (devs[i].status == NT) ? " (untested)" : "");
Uwe Hermann515ab3d2009-05-15 17:02:34 +0000137 }
138}