blob: 2afc2bfa62b6810956a69da7bd730f59b9f93722 [file] [log] [blame]
Stefan Reinauer6a5bc462007-01-17 10:57:42 +00001/*
Uwe Hermannafe83092007-09-28 15:45:43 +00002 * This file is part of the superiotool project.
Stefan Reinauer6a5bc462007-01-17 10:57:42 +00003 *
4 * Copyright (C) 2006 Ronald Minnich <rminnich@gmail.com>
Uwe Hermannbd263922007-09-01 19:42:42 +00005 * Copyright (C) 2007 Uwe Hermann <uwe@hermann-uwe.de>
Uwe Hermann519419b2007-09-16 20:59:01 +00006 * Copyright (C) 2007 Carl-Daniel Hailfinger
Robinson P. Tryon552cfb72008-01-15 22:30:55 +00007 * Copyright (C) 2008 Robinson P. Tryon <bishop.robinson@gmail.com>
Stefan Reinauer6a5bc462007-01-17 10:57:42 +00008 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
22 */
23
Uwe Hermann0120e1a2007-09-16 18:11:03 +000024#include "superiotool.h"
Ronald G. Minnich394e7c42006-02-22 22:12:21 +000025
Andriy Gaponb64aa602008-10-28 22:13:38 +000026#if defined(__FreeBSD__)
27#include <fcntl.h>
28#include <unistd.h>
29#endif
30
Uwe Hermann3acf31e2007-09-19 01:55:35 +000031/* Command line options. */
Ronald Hoogenboom0be73bb2008-02-25 22:32:41 +000032int dump = 0, verbose = 0, extra_dump = 0;
Uwe Hermann3acf31e2007-09-19 01:55:35 +000033
Uwe Hermanne9d46162007-10-07 20:01:23 +000034/* Global flag which indicates whether a chip was detected at all. */
35int chip_found = 0;
36
Uwe Hermannde24a0e2007-09-19 00:03:14 +000037uint8_t regval(uint16_t port, uint8_t reg)
Uwe Hermann2046ff92007-09-01 21:02:44 +000038{
Andriy Gaponb64aa602008-10-28 22:13:38 +000039 OUTB(reg, port);
40 return INB(port + ((port == 0x3bd) ? 2 : 1)); /* 0x3bd is special. */
Ronald G. Minnich394e7c42006-02-22 22:12:21 +000041}
42
Uwe Hermannde24a0e2007-09-19 00:03:14 +000043void regwrite(uint16_t port, uint8_t reg, uint8_t val)
Uwe Hermann2046ff92007-09-01 21:02:44 +000044{
Andriy Gaponb64aa602008-10-28 22:13:38 +000045 OUTB(reg, port);
46 OUTB(val, port + 1);
Carl-Daniel Hailfinger7a7890a2007-08-27 07:28:28 +000047}
48
Uwe Hermannb4db2202007-09-20 23:37:56 +000049void enter_conf_mode_winbond_fintek_ite_8787(uint16_t port)
50{
Andriy Gaponb64aa602008-10-28 22:13:38 +000051 OUTB(0x87, port);
52 OUTB(0x87, port);
Uwe Hermannb4db2202007-09-20 23:37:56 +000053}
54
55void exit_conf_mode_winbond_fintek_ite_8787(uint16_t port)
56{
Andriy Gaponb64aa602008-10-28 22:13:38 +000057 OUTB(0xaa, port); /* Fintek, Winbond */
Uwe Hermannb4db2202007-09-20 23:37:56 +000058 regwrite(port, 0x02, 0x02); /* ITE */
59}
60
Uwe Hermann7e7e9ac2007-09-19 15:52:23 +000061int superio_unknown(const struct superio_registers reg_table[], uint16_t id)
62{
63 return !strncmp(get_superio_name(reg_table, id), "<unknown>", 9);
64}
65
Uwe Hermann3acf31e2007-09-19 01:55:35 +000066const char *get_superio_name(const struct superio_registers reg_table[],
67 uint16_t id)
68{
69 int i;
70
71 for (i = 0; /* Nothing */; i++) {
72 if (reg_table[i].superio_id == EOT)
73 break;
74
75 if ((uint16_t)reg_table[i].superio_id != id)
76 continue;
77
78 return reg_table[i].name;
79 }
80
81 return "<unknown>";
82}
83
Uwe Hermann2c290e32007-09-20 00:00:49 +000084static void dump_regs(const struct superio_registers reg_table[],
Stefan Reinauer7a51e502008-12-01 14:18:57 +000085 int i, int j, uint16_t port, uint8_t ldn_sel)
Uwe Hermann2c290e32007-09-20 00:00:49 +000086{
Ward Vandewege2ee78d82007-09-24 22:02:31 +000087 int k;
Uwe Hermann42cccdf2008-03-29 01:35:21 +000088 const int16_t *idx;
Uwe Hermann2c290e32007-09-20 00:00:49 +000089
90 if (reg_table[i].ldn[j].ldn != NOLDN) {
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +000091 printf("LDN 0x%02x", reg_table[i].ldn[j].ldn);
Uwe Hermann2c290e32007-09-20 00:00:49 +000092 if (reg_table[i].ldn[j].name != NULL)
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +000093 printf(" (%s)", reg_table[i].ldn[j].name);
Stefan Reinauer7a51e502008-12-01 14:18:57 +000094 regwrite(port, ldn_sel, reg_table[i].ldn[j].ldn);
Uwe Hermann2c290e32007-09-20 00:00:49 +000095 } else {
96 printf("Register dump:");
97 }
98
99 idx = reg_table[i].ldn[j].idx;
100
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000101 printf("\nidx");
102 for (k = 0; idx[k] != EOT; k++) {
103 if (k && !(k % 8))
104 putchar(' ');
105 printf(" %02x", idx[k]);
Uwe Hermann2c290e32007-09-20 00:00:49 +0000106 }
107
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000108 printf("\nval");
109 for (k = 0; idx[k] != EOT; k++) {
110 if (k && !(k % 8))
111 putchar(' ');
112 printf(" %02x", regval(port, idx[k]));
Uwe Hermann2c290e32007-09-20 00:00:49 +0000113 }
114
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000115 printf("\ndef");
Uwe Hermann2c290e32007-09-20 00:00:49 +0000116 idx = reg_table[i].ldn[j].def;
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000117 for (k = 0; idx[k] != EOT; k++) {
118 if (k && !(k % 8))
119 putchar(' ');
120 if (idx[k] == NANA)
121 printf(" NA");
Uwe Hermann2c290e32007-09-20 00:00:49 +0000122 else if (idx[k] == RSVD)
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000123 printf(" RR");
Uwe Hermann59b99d92007-11-29 02:43:50 +0000124 else if (idx[k] == MISC)
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000125 printf(" MM");
Uwe Hermann2c290e32007-09-20 00:00:49 +0000126 else
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000127 printf(" %02x", idx[k]);
Uwe Hermann2c290e32007-09-20 00:00:49 +0000128 }
129 printf("\n");
130}
131
132void dump_superio(const char *vendor,
133 const struct superio_registers reg_table[],
Stefan Reinauer7a51e502008-12-01 14:18:57 +0000134 uint16_t port, uint16_t id, uint8_t ldn_sel)
Uwe Hermann519419b2007-09-16 20:59:01 +0000135{
Uwe Hermann2c290e32007-09-20 00:00:49 +0000136 int i, j, no_dump_available = 1;
Uwe Hermann519419b2007-09-16 20:59:01 +0000137
Uwe Hermann7e7e9ac2007-09-19 15:52:23 +0000138 if (!dump)
139 return;
140
Uwe Hermann519419b2007-09-16 20:59:01 +0000141 for (i = 0; /* Nothing */; i++) {
142 if (reg_table[i].superio_id == EOT)
143 break;
144
Uwe Hermannde24a0e2007-09-19 00:03:14 +0000145 if ((uint16_t)reg_table[i].superio_id != id)
Uwe Hermann519419b2007-09-16 20:59:01 +0000146 continue;
147
Uwe Hermannde24a0e2007-09-19 00:03:14 +0000148 for (j = 0; /* Nothing */; j++) {
Uwe Hermann519419b2007-09-16 20:59:01 +0000149 if (reg_table[i].ldn[j].ldn == EOT)
150 break;
Uwe Hermann2c290e32007-09-20 00:00:49 +0000151 no_dump_available = 0;
Stefan Reinauer7a51e502008-12-01 14:18:57 +0000152 dump_regs(reg_table, i, j, port, ldn_sel);
Uwe Hermann519419b2007-09-16 20:59:01 +0000153 }
Uwe Hermann25a6c0f2007-09-19 00:48:42 +0000154
Uwe Hermann2c290e32007-09-20 00:00:49 +0000155 if (no_dump_available)
156 printf("No dump available for this Super I/O\n");
Uwe Hermann519419b2007-09-16 20:59:01 +0000157 }
158}
159
Stefan Reinauer6df0c622009-03-11 14:48:20 +0000160void dump_io(uint16_t iobase, uint16_t length)
161{
162 uint16_t i;
163
Uwe Hermann83da8dc2009-03-25 17:38:40 +0000164 printf("Dumping %d I/O mapped registers at base 0x%04x:\n",
Stefan Reinauer6df0c622009-03-11 14:48:20 +0000165 length, iobase);
Uwe Hermann83da8dc2009-03-25 17:38:40 +0000166 for (i = 0; i < length; i++)
167 printf("%02x ", i);
Stefan Reinauer6df0c622009-03-11 14:48:20 +0000168 printf("\n");
Uwe Hermann83da8dc2009-03-25 17:38:40 +0000169 for (i = 0; i < length; i++)
170 printf("%02x ", INB(iobase + i));
Stefan Reinauer6df0c622009-03-11 14:48:20 +0000171 printf("\n");
172}
173
Uwe Hermann8b8d0392007-10-04 15:23:38 +0000174void probing_for(const char *vendor, const char *info, uint16_t port)
Uwe Hermann2c290e32007-09-20 00:00:49 +0000175{
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000176 if (!verbose)
177 return;
178
Uwe Hermann8b8d0392007-10-04 15:23:38 +0000179 /* Yes, there's no space between '%s' and 'at'! */
Uwe Hermanneec5ff42008-03-01 18:49:39 +0000180 printf("Probing for %s Super I/O %sat 0x%x...\n", vendor, info, port);
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000181}
182
Robinson P. Tryon552cfb72008-01-15 22:30:55 +0000183/** Print a list of all supported chips from the given vendor. */
184void print_vendor_chips(const char *vendor,
185 const struct superio_registers reg_table[])
186{
187 int i;
188
189 for (i = 0; reg_table[i].superio_id != EOT; i++) {
190 printf("%s %s", vendor, reg_table[i].name);
191
192 /* Unless the ldn is empty, assume this chip has a dump. */
193 if (reg_table[i].ldn[0].ldn != EOT)
194 printf(" (dump available)");
195
196 printf("\n");
197 }
198
199 /* If we printed any chips for this vendor, put in a blank line. */
200 if (i != 0)
201 printf("\n");
202}
203
204/** Print a list of all chips supported by superiotool. */
205void print_list_of_supported_chips(void)
206{
207 int i;
208
209 printf("Supported Super I/O chips:\n\n");
210
211 for (i = 0; i < ARRAY_SIZE(vendor_print_functions); i++)
212 vendor_print_functions[i].print_list();
213
214 printf("See <http://coreboot.org/Superiotool#Supported_devices> "
215 "for more information.\n");
216}
217
Robinson P. Tryonec1edd12007-10-02 23:32:21 +0000218static void print_version(void)
219{
Ulf Jordan39a5bf72007-10-13 18:06:12 +0000220 printf("superiotool r%s\n", SUPERIOTOOL_VERSION);
Robinson P. Tryonec1edd12007-10-02 23:32:21 +0000221}
222
Uwe Hermann2046ff92007-09-01 21:02:44 +0000223int main(int argc, char *argv[])
Stefan Reinauer6a5bc462007-01-17 10:57:42 +0000224{
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000225 int i, j, opt, option_index;
Andriy Gaponb64aa602008-10-28 22:13:38 +0000226#if defined(__FreeBSD__)
227 int io_fd;
228#endif
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000229
Uwe Hermann246be7d2007-10-31 22:22:11 +0000230 static const struct option long_options[] = {
Uwe Hermanneddc4732007-09-20 23:57:44 +0000231 {"dump", no_argument, NULL, 'd'},
Ronald Hoogenboom0be73bb2008-02-25 22:32:41 +0000232 {"extra-dump", no_argument, NULL, 'e'},
Robinson P. Tryon552cfb72008-01-15 22:30:55 +0000233 {"list-supported", no_argument, NULL, 'l'},
Uwe Hermanneddc4732007-09-20 23:57:44 +0000234 {"verbose", no_argument, NULL, 'V'},
235 {"version", no_argument, NULL, 'v'},
236 {"help", no_argument, NULL, 'h'},
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000237 {0, 0, 0, 0}
238 };
239
Ronald Hoogenboom0be73bb2008-02-25 22:32:41 +0000240 while ((opt = getopt_long(argc, argv, "delVvh",
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000241 long_options, &option_index)) != EOF) {
242 switch (opt) {
243 case 'd':
244 dump = 1;
245 break;
Ronald Hoogenboom0be73bb2008-02-25 22:32:41 +0000246 case 'e':
247 extra_dump = 1;
248 break;
Robinson P. Tryon552cfb72008-01-15 22:30:55 +0000249 case 'l':
250 print_list_of_supported_chips();
251 exit(0);
252 break;
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000253 case 'V':
254 verbose = 1;
255 break;
256 case 'v':
Robinson P. Tryonec1edd12007-10-02 23:32:21 +0000257 print_version();
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000258 exit(0);
259 break;
260 case 'h':
Uwe Hermanne4749562007-09-19 16:26:18 +0000261 printf(USAGE);
Uwe Hermann969a9f62008-03-17 13:43:48 +0000262 printf(USAGE_INFO);
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000263 exit(0);
264 break;
265 default:
266 /* Unknown option. */
267 exit(1);
268 break;
269 }
270 }
Uwe Hermannd754d2c2007-09-18 23:30:24 +0000271
Andriy Gaponb64aa602008-10-28 22:13:38 +0000272#if defined(__FreeBSD__)
273 if ((io_fd = open("/dev/io", O_RDWR)) < 0) {
274 perror("/dev/io");
275#else
Ronald G. Minnich394e7c42006-02-22 22:12:21 +0000276 if (iopl(3) < 0) {
277 perror("iopl");
Andriy Gaponb64aa602008-10-28 22:13:38 +0000278#endif
Uwe Hermann7e7e9ac2007-09-19 15:52:23 +0000279 printf("Superiotool must be run as root.\n");
Ronald G. Minnich394e7c42006-02-22 22:12:21 +0000280 exit(1);
281 }
282
Uwe Hermannd937b522007-10-17 23:42:02 +0000283 print_version();
284
Carl-Daniel Hailfingerbb38f322010-01-24 01:40:46 +0000285#ifdef PCI_SUPPORT
286 /* Do some basic libpci init. */
287 pacc = pci_alloc();
288 pci_init(pacc);
289 pci_scan_bus(pacc);
290#endif
291
Uwe Hermannd754d2c2007-09-18 23:30:24 +0000292 for (i = 0; i < ARRAY_SIZE(superio_ports_table); i++) {
293 for (j = 0; superio_ports_table[i].ports[j] != EOT; j++)
294 superio_ports_table[i].probe_idregs(
295 superio_ports_table[i].ports[j]);
296 }
Stefan Reinauer6a5bc462007-01-17 10:57:42 +0000297
Uwe Hermanne9d46162007-10-07 20:01:23 +0000298 if (!chip_found)
299 printf("No Super I/O found\n");
300
Andriy Gaponb64aa602008-10-28 22:13:38 +0000301#if defined(__FreeBSD__)
302 close(io_fd);
303#endif
Stefan Reinauer6a5bc462007-01-17 10:57:42 +0000304 return 0;
Ronald G. Minnich394e7c42006-02-22 22:12:21 +0000305}