blob: 72e6b6c923e097608e665896a77bd5e2f8f27efb [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
Stefan Reinauer6a5bc462007-01-17 10:57:42 +00007 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
21 */
22
Uwe Hermann0120e1a2007-09-16 18:11:03 +000023#include "superiotool.h"
Ronald G. Minnich394e7c42006-02-22 22:12:21 +000024
Uwe Hermann3acf31e2007-09-19 01:55:35 +000025/* Command line options. */
Uwe Hermann7dabe5e2007-10-17 23:43:59 +000026int dump = 0, verbose = 0;
Uwe Hermann3acf31e2007-09-19 01:55:35 +000027
Uwe Hermanne9d46162007-10-07 20:01:23 +000028/* Global flag which indicates whether a chip was detected at all. */
29int chip_found = 0;
30
Uwe Hermannde24a0e2007-09-19 00:03:14 +000031uint8_t regval(uint16_t port, uint8_t reg)
Uwe Hermann2046ff92007-09-01 21:02:44 +000032{
Ronald G. Minnich394e7c42006-02-22 22:12:21 +000033 outb(reg, port);
Carl-Daniel Hailfingerb1786c22007-08-28 10:43:57 +000034 return inb(port + 1);
Ronald G. Minnich394e7c42006-02-22 22:12:21 +000035}
36
Uwe Hermannde24a0e2007-09-19 00:03:14 +000037void regwrite(uint16_t port, uint8_t reg, uint8_t val)
Uwe Hermann2046ff92007-09-01 21:02:44 +000038{
Carl-Daniel Hailfinger7a7890a2007-08-27 07:28:28 +000039 outb(reg, port);
Carl-Daniel Hailfingerb1786c22007-08-28 10:43:57 +000040 outb(val, port + 1);
Carl-Daniel Hailfinger7a7890a2007-08-27 07:28:28 +000041}
42
Uwe Hermannb4db2202007-09-20 23:37:56 +000043void enter_conf_mode_winbond_fintek_ite_8787(uint16_t port)
44{
45 outb(0x87, port);
46 outb(0x87, port);
47}
48
49void exit_conf_mode_winbond_fintek_ite_8787(uint16_t port)
50{
51 outb(0xaa, port); /* Fintek, Winbond */
52 regwrite(port, 0x02, 0x02); /* ITE */
53}
54
Uwe Hermann7e7e9ac2007-09-19 15:52:23 +000055int superio_unknown(const struct superio_registers reg_table[], uint16_t id)
56{
57 return !strncmp(get_superio_name(reg_table, id), "<unknown>", 9);
58}
59
Uwe Hermann3acf31e2007-09-19 01:55:35 +000060const char *get_superio_name(const struct superio_registers reg_table[],
61 uint16_t id)
62{
63 int i;
64
65 for (i = 0; /* Nothing */; i++) {
66 if (reg_table[i].superio_id == EOT)
67 break;
68
69 if ((uint16_t)reg_table[i].superio_id != id)
70 continue;
71
72 return reg_table[i].name;
73 }
74
75 return "<unknown>";
76}
77
Uwe Hermann2c290e32007-09-20 00:00:49 +000078static void dump_regs(const struct superio_registers reg_table[],
79 int i, int j, uint16_t port)
80{
Ward Vandewege2ee78d82007-09-24 22:02:31 +000081 int k;
82 const int *idx;
Uwe Hermann2c290e32007-09-20 00:00:49 +000083
84 if (reg_table[i].ldn[j].ldn != NOLDN) {
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +000085 printf("LDN 0x%02x", reg_table[i].ldn[j].ldn);
Uwe Hermann2c290e32007-09-20 00:00:49 +000086 if (reg_table[i].ldn[j].name != NULL)
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +000087 printf(" (%s)", reg_table[i].ldn[j].name);
Uwe Hermann2c290e32007-09-20 00:00:49 +000088 regwrite(port, 0x07, reg_table[i].ldn[j].ldn);
89 } else {
90 printf("Register dump:");
91 }
92
93 idx = reg_table[i].ldn[j].idx;
94
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +000095 printf("\nidx");
96 for (k = 0; idx[k] != EOT; k++) {
97 if (k && !(k % 8))
98 putchar(' ');
99 printf(" %02x", idx[k]);
Uwe Hermann2c290e32007-09-20 00:00:49 +0000100 }
101
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000102 printf("\nval");
103 for (k = 0; idx[k] != EOT; k++) {
104 if (k && !(k % 8))
105 putchar(' ');
106 printf(" %02x", regval(port, idx[k]));
Uwe Hermann2c290e32007-09-20 00:00:49 +0000107 }
108
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000109 printf("\ndef");
Uwe Hermann2c290e32007-09-20 00:00:49 +0000110 idx = reg_table[i].ldn[j].def;
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000111 for (k = 0; idx[k] != EOT; k++) {
112 if (k && !(k % 8))
113 putchar(' ');
114 if (idx[k] == NANA)
115 printf(" NA");
Uwe Hermann2c290e32007-09-20 00:00:49 +0000116 else if (idx[k] == RSVD)
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000117 printf(" RR");
Uwe Hermann2c290e32007-09-20 00:00:49 +0000118 else if (idx[k] == MISC) /* TODO */
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000119 printf(" MM");
Uwe Hermann2c290e32007-09-20 00:00:49 +0000120 else
Frieder Ferlemanna422c2d2007-11-13 09:09:33 +0000121 printf(" %02x", idx[k]);
Uwe Hermann2c290e32007-09-20 00:00:49 +0000122 }
123 printf("\n");
124}
125
126void dump_superio(const char *vendor,
127 const struct superio_registers reg_table[],
Uwe Hermannde24a0e2007-09-19 00:03:14 +0000128 uint16_t port, uint16_t id)
Uwe Hermann519419b2007-09-16 20:59:01 +0000129{
Uwe Hermann2c290e32007-09-20 00:00:49 +0000130 int i, j, no_dump_available = 1;
Uwe Hermann519419b2007-09-16 20:59:01 +0000131
Uwe Hermann7e7e9ac2007-09-19 15:52:23 +0000132 if (!dump)
133 return;
134
Uwe Hermann519419b2007-09-16 20:59:01 +0000135 for (i = 0; /* Nothing */; i++) {
136 if (reg_table[i].superio_id == EOT)
137 break;
138
Uwe Hermannde24a0e2007-09-19 00:03:14 +0000139 if ((uint16_t)reg_table[i].superio_id != id)
Uwe Hermann519419b2007-09-16 20:59:01 +0000140 continue;
141
Uwe Hermannde24a0e2007-09-19 00:03:14 +0000142 for (j = 0; /* Nothing */; j++) {
Uwe Hermann519419b2007-09-16 20:59:01 +0000143 if (reg_table[i].ldn[j].ldn == EOT)
144 break;
Uwe Hermann2c290e32007-09-20 00:00:49 +0000145 no_dump_available = 0;
146 dump_regs(reg_table, i, j, port);
Uwe Hermann519419b2007-09-16 20:59:01 +0000147 }
Uwe Hermann25a6c0f2007-09-19 00:48:42 +0000148
Uwe Hermann2c290e32007-09-20 00:00:49 +0000149 if (no_dump_available)
150 printf("No dump available for this Super I/O\n");
Uwe Hermann519419b2007-09-16 20:59:01 +0000151 }
152}
153
Uwe Hermann8b8d0392007-10-04 15:23:38 +0000154void probing_for(const char *vendor, const char *info, uint16_t port)
Uwe Hermann2c290e32007-09-20 00:00:49 +0000155{
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000156 if (!verbose)
157 return;
158
Uwe Hermann8b8d0392007-10-04 15:23:38 +0000159 /* Yes, there's no space between '%s' and 'at'! */
160 printf("Probing for %s Super I/O %sat 0x%x...\n",
161 vendor, info, port);
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000162}
163
Robinson P. Tryonec1edd12007-10-02 23:32:21 +0000164static void print_version(void)
165{
Ulf Jordan39a5bf72007-10-13 18:06:12 +0000166 printf("superiotool r%s\n", SUPERIOTOOL_VERSION);
Robinson P. Tryonec1edd12007-10-02 23:32:21 +0000167}
168
Uwe Hermann2046ff92007-09-01 21:02:44 +0000169int main(int argc, char *argv[])
Stefan Reinauer6a5bc462007-01-17 10:57:42 +0000170{
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000171 int i, j, opt, option_index;
172
Uwe Hermann246be7d2007-10-31 22:22:11 +0000173 static const struct option long_options[] = {
Uwe Hermanneddc4732007-09-20 23:57:44 +0000174 {"dump", no_argument, NULL, 'd'},
Uwe Hermanneddc4732007-09-20 23:57:44 +0000175 {"verbose", no_argument, NULL, 'V'},
176 {"version", no_argument, NULL, 'v'},
177 {"help", no_argument, NULL, 'h'},
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000178 {0, 0, 0, 0}
179 };
180
Uwe Hermann7dabe5e2007-10-17 23:43:59 +0000181 while ((opt = getopt_long(argc, argv, "dVvh",
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000182 long_options, &option_index)) != EOF) {
183 switch (opt) {
184 case 'd':
185 dump = 1;
186 break;
187 case 'V':
188 verbose = 1;
189 break;
190 case 'v':
Robinson P. Tryonec1edd12007-10-02 23:32:21 +0000191 print_version();
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000192 exit(0);
193 break;
194 case 'h':
Uwe Hermanne4749562007-09-19 16:26:18 +0000195 printf(USAGE);
Uwe Hermann3acf31e2007-09-19 01:55:35 +0000196 exit(0);
197 break;
198 default:
199 /* Unknown option. */
200 exit(1);
201 break;
202 }
203 }
Uwe Hermannd754d2c2007-09-18 23:30:24 +0000204
Ronald G. Minnich394e7c42006-02-22 22:12:21 +0000205 if (iopl(3) < 0) {
206 perror("iopl");
Uwe Hermann7e7e9ac2007-09-19 15:52:23 +0000207 printf("Superiotool must be run as root.\n");
Ronald G. Minnich394e7c42006-02-22 22:12:21 +0000208 exit(1);
209 }
210
Uwe Hermannd937b522007-10-17 23:42:02 +0000211 print_version();
212
Uwe Hermannd754d2c2007-09-18 23:30:24 +0000213 for (i = 0; i < ARRAY_SIZE(superio_ports_table); i++) {
214 for (j = 0; superio_ports_table[i].ports[j] != EOT; j++)
215 superio_ports_table[i].probe_idregs(
216 superio_ports_table[i].ports[j]);
217 }
Stefan Reinauer6a5bc462007-01-17 10:57:42 +0000218
Uwe Hermanne9d46162007-10-07 20:01:23 +0000219 if (!chip_found)
220 printf("No Super I/O found\n");
221
Stefan Reinauer6a5bc462007-01-17 10:57:42 +0000222 return 0;
Ronald G. Minnich394e7c42006-02-22 22:12:21 +0000223}