| 1 | /* |
|---|
| 2 | * This file is part of the flashrom project. |
|---|
| 3 | * |
|---|
| 4 | * Copyright (C) 2009 Uwe Hermann <uwe@hermann-uwe.de> |
|---|
| 5 | * Copyright (C) 2009 Carl-Daniel Hailfinger |
|---|
| 6 | * |
|---|
| 7 | * This program is free software; you can redistribute it and/or modify |
|---|
| 8 | * it under the terms of the GNU General Public License as published by |
|---|
| 9 | * the Free Software Foundation; either version 2 of the License, or |
|---|
| 10 | * (at your option) any later version. |
|---|
| 11 | * |
|---|
| 12 | * This program is distributed in the hope that it will be useful, |
|---|
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
|---|
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|---|
| 15 | * GNU General Public License for more details. |
|---|
| 16 | * |
|---|
| 17 | * You should have received a copy of the GNU General Public License |
|---|
| 18 | * along with this program; if not, write to the Free Software |
|---|
| 19 | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
|---|
| 20 | */ |
|---|
| 21 | |
|---|
| 22 | #include <stdio.h> |
|---|
| 23 | #include <string.h> |
|---|
| 24 | #include <stdlib.h> |
|---|
| 25 | #include <time.h> |
|---|
| 26 | #include "flash.h" |
|---|
| 27 | #include "flashchips.h" |
|---|
| 28 | #include "programmer.h" |
|---|
| 29 | |
|---|
| 30 | static const char wiki_header[] = "= Supported devices =\n\n\ |
|---|
| 31 | <div style=\"margin-top:0.5em; padding:0.5em 0.5em 0.5em 0.5em; \ |
|---|
| 32 | background-color:#eeeeee; text-align:right; border:1px solid #aabbcc;\">\ |
|---|
| 33 | <small>\n\ |
|---|
| 34 | Please do '''not''' edit these tables in the wiki directly, they are \ |
|---|
| 35 | generated by pasting '''flashrom -z''' output.<br />\ |
|---|
| 36 | '''Last update:''' %s(generated by flashrom %s)\n</small></div>\n"; |
|---|
| 37 | |
|---|
| 38 | #if CONFIG_INTERNAL == 1 |
|---|
| 39 | static const char chipset_th[] = "{| border=\"0\" style=\"font-size: smaller\"\n\ |
|---|
| 40 | |- bgcolor=\"#6699dd\"\n! align=\"left\" | Vendor\n\ |
|---|
| 41 | ! align=\"left\" | Southbridge\n! align=\"center\" | PCI IDs\n\ |
|---|
| 42 | ! align=\"center\" | Status\n\n"; |
|---|
| 43 | |
|---|
| 44 | static const char board_th[] = "{| border=\"0\" style=\"font-size: smaller\" \ |
|---|
| 45 | valign=\"top\"\n|- bgcolor=\"#6699dd\"\n! align=\"left\" | Vendor\n\ |
|---|
| 46 | ! align=\"left\" | Mainboard\n! align=\"left\" | Required option\n\ |
|---|
| 47 | ! align=\"center\" | Status\n\n"; |
|---|
| 48 | |
|---|
| 49 | static const char board_intro[] = "\ |
|---|
| 50 | \n== Supported mainboards ==\n\n\ |
|---|
| 51 | In general, it is very likely that flashrom works out of the box even if your \ |
|---|
| 52 | mainboard is not listed below.\n\nThis is a list of mainboards where we have \ |
|---|
| 53 | verified that they either do or do not need any special initialization to \ |
|---|
| 54 | make flashrom work (given flashrom supports the respective chipset and flash \ |
|---|
| 55 | chip), or that they do not yet work at all. If they do not work, support may \ |
|---|
| 56 | or may not be added later.\n\n\ |
|---|
| 57 | Mainboards (or individual revisions) which don't appear in the list may or may \ |
|---|
| 58 | not work (we don't know, someone has to give it a try). Please report any \ |
|---|
| 59 | further verified mainboards on the [[Mailinglist|mailing list]].\n"; |
|---|
| 60 | #endif |
|---|
| 61 | |
|---|
| 62 | static const char chip_th[] = "{\ |
|---|
| 63 | | border=\"0\" style=\"font-size: smaller\" valign=\"top\"\n\ |
|---|
| 64 | |- bgcolor=\"#6699dd\"\n\ |
|---|
| 65 | ! align=\"left\" | Vendor\n\ |
|---|
| 66 | ! align=\"left\" | Device\n\ |
|---|
| 67 | ! align=\"center\" | Size [kB]\n\ |
|---|
| 68 | ! align=\"center\" | Type\n\ |
|---|
| 69 | ! align=\"center\" colspan=\"4\" | Status\n\ |
|---|
| 70 | ! align=\"center\" colspan=\"2\" | Voltage [V]\n\n\ |
|---|
| 71 | |- bgcolor=\"#6699ff\"\n| colspan=\"4\" | \n\ |
|---|
| 72 | | Probe\n| Read\n| Erase\n| Write\n\ |
|---|
| 73 | | align=\"center\" | Min \n| align=\"center\" | Max\n\n"; |
|---|
| 74 | |
|---|
| 75 | static const char programmer_section[] = "\ |
|---|
| 76 | \n== Supported programmers ==\n\nThis is a list \ |
|---|
| 77 | of supported PCI devices flashrom can use as programmer:\n\n{| border=\"0\" \ |
|---|
| 78 | valign=\"top\"\n| valign=\"top\"|\n\n{| border=\"0\" style=\"font-size: \ |
|---|
| 79 | smaller\" valign=\"top\"\n|- bgcolor=\"#6699dd\"\n! align=\"left\" | Vendor\n\ |
|---|
| 80 | ! align=\"left\" | Device\n! align=\"center\" | PCI IDs\n\ |
|---|
| 81 | ! align=\"center\" | Status\n\n"; |
|---|
| 82 | |
|---|
| 83 | #if CONFIG_INTERNAL == 1 |
|---|
| 84 | static const char laptop_intro[] = "\n== Supported laptops/notebooks ==\n\n\ |
|---|
| 85 | In general, flashing laptops is more difficult because laptops\n\n\ |
|---|
| 86 | * often use the flash chip for stuff besides the BIOS,\n\ |
|---|
| 87 | * often have special protection stuff which has to be handled by flashrom,\n\ |
|---|
| 88 | * often use flash translation circuits which need drivers in flashrom.\n\n\ |
|---|
| 89 | <div style=\"margin-top:0.5em; padding:0.5em 0.5em 0.5em 0.5em; \ |
|---|
| 90 | background-color:#ff6666; align:right; border:1px solid #000000;\">\n\ |
|---|
| 91 | '''IMPORTANT:''' At this point we recommend to '''not''' use flashrom on \ |
|---|
| 92 | untested laptops unless you have a means to recover from a flashing that goes \ |
|---|
| 93 | wrong (a working backup flash chip and/or good soldering skills).\n</div>\n"; |
|---|
| 94 | |
|---|
| 95 | static void print_supported_chipsets_wiki(int cols) |
|---|
| 96 | { |
|---|
| 97 | int i, j, enablescount = 0, color = 1; |
|---|
| 98 | const struct penable *e; |
|---|
| 99 | |
|---|
| 100 | for (e = chipset_enables; e->vendor_name != NULL; e++) |
|---|
| 101 | enablescount++; |
|---|
| 102 | |
|---|
| 103 | printf("\n== Supported chipsets ==\n\nTotal amount of supported " |
|---|
| 104 | "chipsets: '''%d'''\n\n{| border=\"0\" valign=\"top\"\n| " |
|---|
| 105 | "valign=\"top\"|\n\n%s", enablescount, chipset_th); |
|---|
| 106 | |
|---|
| 107 | e = chipset_enables; |
|---|
| 108 | for (i = 0, j = 0; e[i].vendor_name != NULL; i++, j++) { |
|---|
| 109 | /* Alternate colors if the vendor changes. */ |
|---|
| 110 | if (i > 0 && strcmp(e[i].vendor_name, e[i - 1].vendor_name)) |
|---|
| 111 | color = !color; |
|---|
| 112 | |
|---|
| 113 | printf("|- bgcolor=\"#%s\"\n| %s || %s " |
|---|
| 114 | "|| %04x:%04x || %s\n", (color) ? "eeeeee" : "dddddd", |
|---|
| 115 | e[i].vendor_name, e[i].device_name, |
|---|
| 116 | e[i].vendor_id, e[i].device_id, |
|---|
| 117 | (e[i].status == OK) ? "{{OK}}" : "{{?3}}"); |
|---|
| 118 | |
|---|
| 119 | /* Split table in 'cols' columns. */ |
|---|
| 120 | if (j >= (enablescount / cols + 1)) { |
|---|
| 121 | printf("\n|}\n\n| valign=\"top\"|\n\n%s", chipset_th); |
|---|
| 122 | j = 0; |
|---|
| 123 | } |
|---|
| 124 | } |
|---|
| 125 | |
|---|
| 126 | printf("\n|}\n\n|}\n"); |
|---|
| 127 | } |
|---|
| 128 | |
|---|
| 129 | static void wiki_helper(const char *devicetype, int cols, |
|---|
| 130 | const struct board_info boards[]) |
|---|
| 131 | { |
|---|
| 132 | int i, j, k = 0, boardcount_good = 0, boardcount_bad = 0, color = 1; |
|---|
| 133 | int num_notes = 0; |
|---|
| 134 | char *notes = calloc(1, 1); |
|---|
| 135 | char tmp[900 + 1]; |
|---|
| 136 | const struct board_match *b = board_matches; |
|---|
| 137 | |
|---|
| 138 | for (i = 0; boards[i].vendor != NULL; i++) { |
|---|
| 139 | if (boards[i].working) |
|---|
| 140 | boardcount_good++; |
|---|
| 141 | else |
|---|
| 142 | boardcount_bad++; |
|---|
| 143 | } |
|---|
| 144 | |
|---|
| 145 | printf("\n\nTotal amount of supported %s: '''%d'''. " |
|---|
| 146 | "Not yet supported (i.e., known-bad): '''%d'''.\n\n" |
|---|
| 147 | "{| border=\"0\" valign=\"top\"\n| valign=\"top\"|\n\n%s", |
|---|
| 148 | devicetype, boardcount_good, boardcount_bad, board_th); |
|---|
| 149 | |
|---|
| 150 | for (i = 0, j = 0; boards[i].vendor != NULL; i++, j++) { |
|---|
| 151 | |
|---|
| 152 | /* Alternate colors if the vendor changes. */ |
|---|
| 153 | if (i > 0 && strcmp(boards[i].vendor, boards[i - 1].vendor)) |
|---|
| 154 | color = !color; |
|---|
| 155 | |
|---|
| 156 | k = 0; |
|---|
| 157 | while ((b[k].vendor_name != NULL) && |
|---|
| 158 | (strcmp(b[k].vendor_name, boards[i].vendor) || |
|---|
| 159 | strcmp(b[k].board_name, boards[i].name))) { |
|---|
| 160 | k++; |
|---|
| 161 | } |
|---|
| 162 | |
|---|
| 163 | printf("|- bgcolor=\"#%s\"\n| %s || %s%s %s%s || %s%s%s%s " |
|---|
| 164 | "|| {{%s}}", (color) ? "eeeeee" : "dddddd", |
|---|
| 165 | boards[i].vendor, |
|---|
| 166 | boards[i].url ? "[" : "", |
|---|
| 167 | boards[i].url ? boards[i].url : "", |
|---|
| 168 | boards[i].name, |
|---|
| 169 | boards[i].url ? "]" : "", |
|---|
| 170 | b[k].lb_vendor ? "-p internal:mainboard=" : "—", |
|---|
| 171 | b[k].lb_vendor ? b[k].lb_vendor : "", |
|---|
| 172 | b[k].lb_vendor ? ":" : "", |
|---|
| 173 | b[k].lb_vendor ? b[k].lb_part : "", |
|---|
| 174 | (boards[i].working) ? "OK" : "No"); |
|---|
| 175 | |
|---|
| 176 | if (boards[i].note) { |
|---|
| 177 | printf("<sup>%d</sup>\n", num_notes + 1); |
|---|
| 178 | snprintf(tmp, sizeof(tmp), "<sup>%d</sup> %s<br />\n", |
|---|
| 179 | 1 + num_notes++, boards[i].note); |
|---|
| 180 | notes = strcat_realloc(notes, tmp); |
|---|
| 181 | } else { |
|---|
| 182 | printf("\n"); |
|---|
| 183 | } |
|---|
| 184 | |
|---|
| 185 | /* Split table in 'cols' columns. */ |
|---|
| 186 | if (j >= ((boardcount_good + boardcount_bad) / cols + 1)) { |
|---|
| 187 | printf("\n|}\n\n| valign=\"top\"|\n\n%s", board_th); |
|---|
| 188 | j = 0; |
|---|
| 189 | } |
|---|
| 190 | } |
|---|
| 191 | |
|---|
| 192 | printf("\n|}\n\n|}\n"); |
|---|
| 193 | |
|---|
| 194 | if (num_notes > 0) |
|---|
| 195 | printf("\n<small>\n%s</small>\n", notes); |
|---|
| 196 | free(notes); |
|---|
| 197 | } |
|---|
| 198 | |
|---|
| 199 | static void print_supported_boards_wiki(void) |
|---|
| 200 | { |
|---|
| 201 | printf("%s", board_intro); |
|---|
| 202 | wiki_helper("boards", 2, boards_known); |
|---|
| 203 | |
|---|
| 204 | printf("%s", laptop_intro); |
|---|
| 205 | wiki_helper("laptops", 1, laptops_known); |
|---|
| 206 | } |
|---|
| 207 | #endif |
|---|
| 208 | |
|---|
| 209 | static void print_supported_chips_wiki(int cols) |
|---|
| 210 | { |
|---|
| 211 | int i = 0, c = 1, chipcount = 0; |
|---|
| 212 | const struct flashchip *f, *old = NULL; |
|---|
| 213 | uint32_t t; |
|---|
| 214 | char *s; |
|---|
| 215 | char vmax[6]; |
|---|
| 216 | char vmin[6]; |
|---|
| 217 | |
|---|
| 218 | for (f = flashchips; f->name != NULL; f++) { |
|---|
| 219 | /* Don't count generic entries. */ |
|---|
| 220 | if (!strncmp(f->vendor, "Unknown", 7) || |
|---|
| 221 | !strncmp(f->vendor, "Programmer", 10) || |
|---|
| 222 | !strncmp(f->name, "unknown", 7)) |
|---|
| 223 | continue; |
|---|
| 224 | chipcount++; |
|---|
| 225 | } |
|---|
| 226 | |
|---|
| 227 | printf("\n== Supported chips ==\n\nTotal amount of supported " |
|---|
| 228 | "chips: '''%d'''\n\n{| border=\"0\" valign=\"top\"\n" |
|---|
| 229 | "| valign=\"top\"|\n\n%s", chipcount, chip_th); |
|---|
| 230 | |
|---|
| 231 | for (f = flashchips; f->name != NULL; f++, i++) { |
|---|
| 232 | /* Don't print generic entries. */ |
|---|
| 233 | if (!strncmp(f->vendor, "Unknown", 7) || |
|---|
| 234 | !strncmp(f->vendor, "Programmer", 10) || |
|---|
| 235 | !strncmp(f->name, "unknown", 7)) |
|---|
| 236 | continue; |
|---|
| 237 | |
|---|
| 238 | /* Alternate colors if the vendor changes. */ |
|---|
| 239 | if (old != NULL && strcmp(old->vendor, f->vendor)) |
|---|
| 240 | c = !c; |
|---|
| 241 | |
|---|
| 242 | t = f->tested; |
|---|
| 243 | s = flashbuses_to_text(f->bustype); |
|---|
| 244 | sprintf(vmin, "%0.03f", f->voltage.min / (double)1000); |
|---|
| 245 | sprintf(vmax, "%0.03f", f->voltage.max / (double)1000); |
|---|
| 246 | /* '{{%s}}' is used in combination with 'OK', 'No' and '?3' to |
|---|
| 247 | * select special formatting templates for the bg color. */ |
|---|
| 248 | printf("|- bgcolor=\"#%s\"\n| %s || %s || align=\"right\" | %d " |
|---|
| 249 | "|| %s || {{%s}} || {{%s}} || {{%s}} || {{%s}}" |
|---|
| 250 | "|| %s || %s \n", |
|---|
| 251 | (c == 1) ? "eeeeee" : "dddddd", f->vendor, f->name, |
|---|
| 252 | f->total_size, s, |
|---|
| 253 | (t & TEST_OK_PROBE) ? "OK" : |
|---|
| 254 | (t & TEST_BAD_PROBE) ? "No" : "?3", |
|---|
| 255 | (t & TEST_OK_READ) ? "OK" : |
|---|
| 256 | (t & TEST_BAD_READ) ? "No" : "?3", |
|---|
| 257 | (t & TEST_OK_ERASE) ? "OK" : |
|---|
| 258 | (t & TEST_BAD_ERASE) ? "No" : "?3", |
|---|
| 259 | (t & TEST_OK_WRITE) ? "OK" : |
|---|
| 260 | (t & TEST_BAD_WRITE) ? "No" : "?3", |
|---|
| 261 | f->voltage.min ? vmin : "N/A", |
|---|
| 262 | f->voltage.min ? vmax : "N/A"); |
|---|
| 263 | free(s); |
|---|
| 264 | |
|---|
| 265 | /* Split table into 'cols' columns. */ |
|---|
| 266 | if (i >= (chipcount / cols + 1)) { |
|---|
| 267 | printf("\n|}\n\n| valign=\"top\"|\n\n%s", chip_th); |
|---|
| 268 | i = 0; |
|---|
| 269 | } |
|---|
| 270 | |
|---|
| 271 | old = f; |
|---|
| 272 | } |
|---|
| 273 | |
|---|
| 274 | printf("\n|}\n\n|}\n"); |
|---|
| 275 | } |
|---|
| 276 | |
|---|
| 277 | /* Not needed for CONFIG_INTERNAL, but for all other PCI-based programmers. */ |
|---|
| 278 | #if CONFIG_NIC3COM+CONFIG_NICREALTEK+CONFIG_NICNATSEMI+CONFIG_GFXNVIDIA+CONFIG_DRKAISER+CONFIG_SATASII+CONFIG_ATAHPT+CONFIG_NICINTEL+CONFIG_NICINTEL_SPI+CONFIG_OGP_SPI+CONFIG_SATAMV >= 1 |
|---|
| 279 | static void print_supported_pcidevs_wiki(const struct pcidev_status *devs) |
|---|
| 280 | { |
|---|
| 281 | int i = 0; |
|---|
| 282 | static int c = 0; |
|---|
| 283 | |
|---|
| 284 | /* Alternate colors if the vendor changes. */ |
|---|
| 285 | c = !c; |
|---|
| 286 | |
|---|
| 287 | for (i = 0; devs[i].vendor_name != NULL; i++) { |
|---|
| 288 | printf("|- bgcolor=\"#%s\"\n| %s || %s || " |
|---|
| 289 | "%04x:%04x || {{%s}}\n", (c) ? "eeeeee" : "dddddd", |
|---|
| 290 | devs[i].vendor_name, devs[i].device_name, |
|---|
| 291 | devs[i].vendor_id, devs[i].device_id, |
|---|
| 292 | (devs[i].status == NT) ? "?3" : "OK"); |
|---|
| 293 | } |
|---|
| 294 | } |
|---|
| 295 | #endif |
|---|
| 296 | |
|---|
| 297 | void print_supported_wiki(void) |
|---|
| 298 | { |
|---|
| 299 | time_t t = time(NULL); |
|---|
| 300 | |
|---|
| 301 | printf(wiki_header, ctime(&t), flashrom_version); |
|---|
| 302 | print_supported_chips_wiki(2); |
|---|
| 303 | #if CONFIG_INTERNAL == 1 |
|---|
| 304 | print_supported_chipsets_wiki(3); |
|---|
| 305 | print_supported_boards_wiki(); |
|---|
| 306 | #endif |
|---|
| 307 | printf("%s", programmer_section); |
|---|
| 308 | #if CONFIG_NIC3COM == 1 |
|---|
| 309 | print_supported_pcidevs_wiki(nics_3com); |
|---|
| 310 | #endif |
|---|
| 311 | #if CONFIG_NICREALTEK == 1 |
|---|
| 312 | print_supported_pcidevs_wiki(nics_realtek); |
|---|
| 313 | #endif |
|---|
| 314 | #if CONFIG_NICNATSEMI == 1 |
|---|
| 315 | print_supported_pcidevs_wiki(nics_natsemi); |
|---|
| 316 | #endif |
|---|
| 317 | #if CONFIG_GFXNVIDIA == 1 |
|---|
| 318 | print_supported_pcidevs_wiki(gfx_nvidia); |
|---|
| 319 | #endif |
|---|
| 320 | #if CONFIG_DRKAISER == 1 |
|---|
| 321 | print_supported_pcidevs_wiki(drkaiser_pcidev); |
|---|
| 322 | #endif |
|---|
| 323 | #if CONFIG_SATASII == 1 |
|---|
| 324 | print_supported_pcidevs_wiki(satas_sii); |
|---|
| 325 | #endif |
|---|
| 326 | #if CONFIG_ATAHPT == 1 |
|---|
| 327 | print_supported_pcidevs_wiki(ata_hpt); |
|---|
| 328 | #endif |
|---|
| 329 | #if CONFIG_NICINTEL == 1 |
|---|
| 330 | print_supported_pcidevs_wiki(nics_intel); |
|---|
| 331 | #endif |
|---|
| 332 | #if CONFIG_NICINTEL_SPI == 1 |
|---|
| 333 | print_supported_pcidevs_wiki(nics_intel_spi); |
|---|
| 334 | #endif |
|---|
| 335 | #if CONFIG_OGP_SPI == 1 |
|---|
| 336 | print_supported_pcidevs_wiki(ogp_spi); |
|---|
| 337 | #endif |
|---|
| 338 | #if CONFIG_SATAMV == 1 |
|---|
| 339 | print_supported_pcidevs_wiki(satas_mv); |
|---|
| 340 | #endif |
|---|
| 341 | printf("\n|}\n"); |
|---|
| 342 | } |
|---|
| 343 | |
|---|