Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 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. |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 16 | */ |
| 17 | |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 18 | #include <stdio.h> |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 19 | #include <string.h> |
| 20 | #include <stdlib.h> |
| 21 | #include <time.h> |
| 22 | #include "flash.h" |
| 23 | #include "flashchips.h" |
Carl-Daniel Hailfinger | 5b997c3 | 2010-07-27 22:41:39 +0000 | [diff] [blame] | 24 | #include "programmer.h" |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 25 | |
Mathias Krause | a60faab | 2011-01-17 07:50:42 +0000 | [diff] [blame] | 26 | static const char wiki_header[] = "= Supported devices =\n\n\ |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 27 | <div style=\"margin-top:0.5em; padding:0.5em 0.5em 0.5em 0.5em; \ |
Stefan Tauner | 05aa0f4 | 2014-06-02 00:45:57 +0000 | [diff] [blame] | 28 | background-color:#eeeeee; text-align:left; border:1px solid #aabbcc;\">\ |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 29 | <small>\n\ |
Stefan Tauner | 2a10e70 | 2014-06-02 00:46:02 +0000 | [diff] [blame] | 30 | '''Last update:''' %s (generated by flashrom %s)<br />\n\ |
Stefan Tauner | 05aa0f4 | 2014-06-02 00:45:57 +0000 | [diff] [blame] | 31 | The tables below are generated from flashrom's source by copying the output of '''flashrom -z'''.<br /><br />\n\ |
| 32 | A short explanation of the cells representing the support state follows:<br />\n\ |
| 33 | {| border=\"0\" valign=\"top\"\n\ |
| 34 | ! style=\"text-align:left;\" |\n\ |
| 35 | ! style=\"text-align:left;\" |\n\ |
| 36 | |-\n\ |
| 37 | |{{OK}}\n\ |
| 38 | | The feature was '''tested and should work''' in general unless there is a bug in flashrom or another component in \ |
| 39 | the system prohibits some functionality.\n\ |
| 40 | |-\n\ |
| 41 | |{{Dep}}\n\ |
| 42 | | '''Configuration-dependent'''. The feature was tested and should work in general but there are common \ |
| 43 | configurations that drastically limit flashrom's capabilities or make it completely stop working.\n\ |
| 44 | |-\n\ |
| 45 | |{{?3}}\n\ |
| 46 | | The feature is '''untested''' but believed to be working.\n\ |
| 47 | |-\n\ |
| 48 | |{{NA}}\n\ |
| 49 | | The feature is '''not applicable''' in this configuration (e.g. write operations on ROM chips).\n\ |
| 50 | |-\n\ |
| 51 | |{{No}}\n\ |
| 52 | | The feature is '''known to not work'''. Don't bother testing (nor reporting. Patches welcome! ;).\n\ |
| 53 | |}\n\ |
| 54 | </small></div>\n"; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 55 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 56 | static const char th_start[] = "| valign=\"top\"|\n\n\ |
| 57 | {| border=\"0\" style=\"font-size: smaller\" valign=\"top\"\n\ |
| 58 | |- bgcolor=\"#6699dd\"\n"; |
| 59 | |
Carl-Daniel Hailfinger | 7112772 | 2010-05-31 15:27:27 +0000 | [diff] [blame] | 60 | #if CONFIG_INTERNAL == 1 |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 61 | static const char chipset_th[] = "\ |
| 62 | ! align=\"left\" | Vendor\n\ |
| 63 | ! align=\"left\" | Southbridge\n\ |
| 64 | ! align=\"center\" | PCI IDs\n\ |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 65 | ! align=\"center\" | Status\n\n"; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 66 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 67 | static const char board_th[] = "\ |
| 68 | ! align=\"left\" | Vendor\n\ |
| 69 | ! align=\"left\" | Mainboard\n\ |
| 70 | ! align=\"left\" | Required option\n\ |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 71 | ! align=\"center\" | Status\n\n"; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 72 | |
Mathias Krause | a60faab | 2011-01-17 07:50:42 +0000 | [diff] [blame] | 73 | static const char board_intro[] = "\ |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 74 | \n== Supported mainboards ==\n\n\ |
| 75 | In general, it is very likely that flashrom works out of the box even if your \ |
| 76 | mainboard is not listed below.\n\nThis is a list of mainboards where we have \ |
| 77 | verified that they either do or do not need any special initialization to \ |
| 78 | make flashrom work (given flashrom supports the respective chipset and flash \ |
| 79 | chip), or that they do not yet work at all. If they do not work, support may \ |
| 80 | or may not be added later.\n\n\ |
Stefan Tauner | d06d941 | 2011-06-12 19:47:55 +0000 | [diff] [blame] | 81 | Mainboards (or individual revisions) which don't appear in the list may or may \ |
| 82 | not work (we don't know, someone has to give it a try). Please report any \ |
| 83 | further verified mainboards on the [[Mailinglist|mailing list]].\n"; |
Carl-Daniel Hailfinger | 8841d3e | 2010-05-15 15:04:37 +0000 | [diff] [blame] | 84 | #endif |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 85 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 86 | static const char chip_th[] = "\ |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 87 | ! align=\"left\" | Vendor\n\ |
| 88 | ! align=\"left\" | Device\n\ |
| 89 | ! align=\"center\" | Size [kB]\n\ |
| 90 | ! align=\"center\" | Type\n\ |
| 91 | ! align=\"center\" colspan=\"4\" | Status\n\ |
| 92 | ! align=\"center\" colspan=\"2\" | Voltage [V]\n\n\ |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 93 | |- bgcolor=\"#6699ff\"\n| colspan=\"4\" | \n\ |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 94 | | Probe\n| Read\n| Erase\n| Write\n\ |
Elyes HAOUAS | ac01baa | 2018-05-28 16:52:21 +0200 | [diff] [blame] | 95 | | align=\"center\" | Min\n| align=\"center\" | Max\n\n"; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 96 | |
Stefan Tauner | 1c356bb | 2013-07-11 13:48:52 +0000 | [diff] [blame] | 97 | static const char chip_intro[] = "\ |
| 98 | \n== Supported flash chips ==\n\n\ |
| 99 | The list below contains all chips that have some kind of explicit support added to flashrom and their last \ |
| 100 | known test status. Newer SPI flash chips might work even without explicit support if they implement SFDP ([\ |
| 101 | http://www.jedec.org/standards-documents/docs/jesd216 Serial Flash Discoverable Parameters - JESD216]). \ |
| 102 | Flashrom will detect this automatically and inform you about it.\n\n\ |
| 103 | The names used below are designed to be as concise as possible and hence contain only the characters \ |
| 104 | describing properties that are relevant to flashrom. Irrelevant characters specify attributes flashrom can not \ |
| 105 | use or even detect by itself (e.g. the physical package) and have no effect on flashrom's operation. They are \ |
| 106 | replaced by dots ('.') functioning as wildcards (like in Regular Expressions) or are completely omitted at the \ |
| 107 | end of a name.\n"; |
| 108 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 109 | static const char programmer_th[] = "\ |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 110 | ! align=\"left\" | Programmer\n\ |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 111 | ! align=\"left\" | Vendor\n\ |
| 112 | ! align=\"left\" | Device\n\ |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 113 | ! align=\"center\" | IDs\n\ |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 114 | ! align=\"center\" | Status\n\n"; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 115 | |
Stefan Tauner | 618d897 | 2014-05-28 13:02:13 +0000 | [diff] [blame] | 116 | /* The output of this module relies on MediaWiki templates to select special formatting styles for table cells |
| 117 | * reflecting the test status of the respective hardware. This functions returns the correct template name for |
| 118 | * the supplied enum test_state. */ |
| 119 | static const char *test_state_to_template(enum test_state test_state) |
| 120 | { |
| 121 | switch (test_state) { |
| 122 | case OK: return "OK"; |
| 123 | case BAD: return "No"; |
| 124 | case NA: return "NA"; |
| 125 | case DEP: return "Dep"; |
| 126 | case NT: |
| 127 | default: return "?3"; |
| 128 | } |
| 129 | } |
| 130 | |
Carl-Daniel Hailfinger | 7112772 | 2010-05-31 15:27:27 +0000 | [diff] [blame] | 131 | #if CONFIG_INTERNAL == 1 |
Stefan Tauner | 74dc73f | 2015-03-01 22:04:38 +0000 | [diff] [blame] | 132 | static const char laptop_intro[] = "\n== Supported mobile devices (laptops, tablets etc.) ==\n\n\ |
| 133 | In general, flashing mobile devices is more difficult because they\n\n\ |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 134 | * often use the flash chip for stuff besides the BIOS,\n\ |
| 135 | * often have special protection stuff which has to be handled by flashrom,\n\ |
| 136 | * often use flash translation circuits which need drivers in flashrom.\n\n\ |
| 137 | <div style=\"margin-top:0.5em; padding:0.5em 0.5em 0.5em 0.5em; \ |
| 138 | background-color:#ff6666; align:right; border:1px solid #000000;\">\n\ |
| 139 | '''IMPORTANT:''' At this point we recommend to '''not''' use flashrom on \ |
Stefan Tauner | 74dc73f | 2015-03-01 22:04:38 +0000 | [diff] [blame] | 140 | untested mobile devices unless you have a means to recover from a flashing that goes \ |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 141 | wrong (a working backup flash chip and/or good soldering skills).\n</div>\n"; |
| 142 | |
Uwe Hermann | a6d9c4b | 2010-05-24 15:28:12 +0000 | [diff] [blame] | 143 | static void print_supported_chipsets_wiki(int cols) |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 144 | { |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 145 | int i; |
| 146 | unsigned int lines_per_col; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 147 | const struct penable *e; |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 148 | int enablescount = 0, color = 1; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 149 | |
| 150 | for (e = chipset_enables; e->vendor_name != NULL; e++) |
| 151 | enablescount++; |
| 152 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 153 | /* +1 to force the resulting number of columns to be < cols */ |
| 154 | lines_per_col = enablescount / cols + ((enablescount%cols) > 0 ? 1 : 0); |
| 155 | |
| 156 | printf("\n== Supported chipsets ==\n\nTotal amount of supported chipsets: '''%d'''\n\n" |
| 157 | "{| border=\"0\" valign=\"top\"\n", enablescount); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 158 | |
| 159 | e = chipset_enables; |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 160 | for (i = 0; e[i].vendor_name != NULL; i++) { |
| 161 | if ((i % lines_per_col) == 0) |
| 162 | printf("%s%s", th_start, chipset_th); |
| 163 | |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 164 | /* Alternate colors if the vendor changes. */ |
| 165 | if (i > 0 && strcmp(e[i].vendor_name, e[i - 1].vendor_name)) |
| 166 | color = !color; |
| 167 | |
Uwe Hermann | a6d9c4b | 2010-05-24 15:28:12 +0000 | [diff] [blame] | 168 | printf("|- bgcolor=\"#%s\"\n| %s || %s " |
Stefan Tauner | 618d897 | 2014-05-28 13:02:13 +0000 | [diff] [blame] | 169 | "|| %04x:%04x || {{%s}}\n", (color) ? "eeeeee" : "dddddd", |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 170 | e[i].vendor_name, e[i].device_name, |
| 171 | e[i].vendor_id, e[i].device_id, |
Stefan Tauner | 618d897 | 2014-05-28 13:02:13 +0000 | [diff] [blame] | 172 | test_state_to_template(e[i].status)); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 173 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 174 | if (((i % lines_per_col) + 1) == lines_per_col) |
| 175 | printf("\n|}\n\n"); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 176 | } |
| 177 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 178 | /* end inner table if it did not fill the last column fully */ |
| 179 | if (((i % lines_per_col)) > 0) |
| 180 | printf("\n|}\n\n"); |
| 181 | printf("\n\n|}\n"); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 182 | } |
| 183 | |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 184 | static void print_supported_boards_wiki_helper(const char *devicetype, int cols, const struct board_info boards[]) |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 185 | { |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 186 | int i, k; |
| 187 | unsigned int boardcount, lines_per_col; |
Stefan Tauner | 2c20b28 | 2012-07-28 19:35:26 +0000 | [diff] [blame] | 188 | unsigned int boardcount_good = 0, boardcount_bad = 0, boardcount_nt = 0; |
| 189 | int num_notes = 0, color = 1; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 190 | char *notes = calloc(1, 1); |
| 191 | char tmp[900 + 1]; |
Carl-Daniel Hailfinger | 97d5b12 | 2011-08-31 16:19:50 +0000 | [diff] [blame] | 192 | const struct board_match *b = board_matches; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 193 | |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 194 | for (i = 0; boards[i].vendor != NULL; i++) { |
Stefan Tauner | 2c20b28 | 2012-07-28 19:35:26 +0000 | [diff] [blame] | 195 | if (boards[i].working == OK) |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 196 | boardcount_good++; |
Stefan Tauner | 2c20b28 | 2012-07-28 19:35:26 +0000 | [diff] [blame] | 197 | else if (boards[i].working == NT) |
| 198 | boardcount_nt++; |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 199 | else |
| 200 | boardcount_bad++; |
| 201 | } |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 202 | boardcount = boardcount_good + boardcount_nt + boardcount_bad; |
| 203 | |
| 204 | /* +1 to force the resulting number of columns to be < cols */ |
| 205 | lines_per_col = boardcount / cols + ((boardcount%cols) > 0 ? 1 : 0); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 206 | |
Stefan Tauner | 1c356bb | 2013-07-11 13:48:52 +0000 | [diff] [blame] | 207 | printf("\n\nTotal amount of known good %s: '''%d'''; " |
Stefan Tauner | 2c20b28 | 2012-07-28 19:35:26 +0000 | [diff] [blame] | 208 | "Untested (e.g. user vanished before testing new code): '''%d'''; " |
| 209 | "Not yet supported (i.e. known-bad): '''%d'''.\n\n" |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 210 | "{| border=\"0\" valign=\"top\"\n", devicetype, boardcount_good, boardcount_nt, boardcount_bad); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 211 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 212 | for (i = 0; boards[i].vendor != NULL; i++) { |
| 213 | if ((i % lines_per_col) == 0) |
| 214 | printf("%s%s", th_start, board_th); |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 215 | |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 216 | /* Alternate colors if the vendor changes. */ |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 217 | if (i > 0 && strcmp(boards[i].vendor, boards[i - 1].vendor)) |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 218 | color = !color; |
| 219 | |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 220 | k = 0; |
Carl-Daniel Hailfinger | 4146ced | 2010-06-07 11:10:43 +0000 | [diff] [blame] | 221 | while ((b[k].vendor_name != NULL) && |
| 222 | (strcmp(b[k].vendor_name, boards[i].vendor) || |
| 223 | strcmp(b[k].board_name, boards[i].name))) { |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 224 | k++; |
| 225 | } |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 226 | |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 227 | printf("|- bgcolor=\"#%s\"\n| %s || %s%s %s%s || %s%s%s%s " |
| 228 | "|| {{%s}}", (color) ? "eeeeee" : "dddddd", |
| 229 | boards[i].vendor, |
| 230 | boards[i].url ? "[" : "", |
| 231 | boards[i].url ? boards[i].url : "", |
| 232 | boards[i].name, |
| 233 | boards[i].url ? "]" : "", |
Carl-Daniel Hailfinger | 2d927fb | 2012-01-04 00:48:27 +0000 | [diff] [blame] | 234 | b[k].lb_vendor ? "-p internal:mainboard=" : "—", |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 235 | b[k].lb_vendor ? b[k].lb_vendor : "", |
| 236 | b[k].lb_vendor ? ":" : "", |
| 237 | b[k].lb_vendor ? b[k].lb_part : "", |
Stefan Tauner | 618d897 | 2014-05-28 13:02:13 +0000 | [diff] [blame] | 238 | test_state_to_template(boards[i].working)); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 239 | |
Peter Lemenkov | 4adf8a6 | 2010-06-01 10:13:17 +0000 | [diff] [blame] | 240 | if (boards[i].note) { |
Stefan Tauner | 1c356bb | 2013-07-11 13:48:52 +0000 | [diff] [blame] | 241 | num_notes++; |
| 242 | printf(" <span id=\"%s_ref%d\"><sup>[[#%s_note%d|%d]]</sup></span>\n", |
| 243 | devicetype, num_notes, devicetype, num_notes, num_notes); |
| 244 | int ret = snprintf(tmp, sizeof(tmp), |
| 245 | "<span id=\"%s_note%d\">%d. [[#%s_ref%d|↑]]</span>" |
| 246 | " <nowiki>%s</nowiki><br />\n", devicetype, num_notes, num_notes, |
| 247 | devicetype, num_notes, boards[i].note); |
| 248 | if (ret < 0 || ret >= sizeof(tmp)) { |
| 249 | fprintf(stderr, "Footnote text #%d of %s truncated (ret=%d, sizeof(tmp)=%zu)\n", |
| 250 | num_notes, devicetype, ret, sizeof(tmp)); |
| 251 | } |
Mathias Krause | a60faab | 2011-01-17 07:50:42 +0000 | [diff] [blame] | 252 | notes = strcat_realloc(notes, tmp); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 253 | } else { |
| 254 | printf("\n"); |
| 255 | } |
| 256 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 257 | if (((i % lines_per_col) + 1) == lines_per_col) |
| 258 | printf("\n|}\n\n"); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 259 | } |
| 260 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 261 | /* end inner table if it did not fill the last column fully */ |
| 262 | if (((i % lines_per_col)) > 0) |
| 263 | printf("\n|}\n\n"); |
Stefan Tauner | 352e50b | 2013-02-22 15:58:45 +0000 | [diff] [blame] | 264 | printf("|}\n"); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 265 | |
| 266 | if (num_notes > 0) |
| 267 | printf("\n<small>\n%s</small>\n", notes); |
| 268 | free(notes); |
| 269 | } |
| 270 | |
Uwe Hermann | a6d9c4b | 2010-05-24 15:28:12 +0000 | [diff] [blame] | 271 | static void print_supported_boards_wiki(void) |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 272 | { |
| 273 | printf("%s", board_intro); |
Stefan Tauner | 74dc73f | 2015-03-01 22:04:38 +0000 | [diff] [blame] | 274 | print_supported_boards_wiki_helper("mainboards", 2, boards_known); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 275 | |
| 276 | printf("%s", laptop_intro); |
Stefan Tauner | 74dc73f | 2015-03-01 22:04:38 +0000 | [diff] [blame] | 277 | print_supported_boards_wiki_helper("mobile devices", 1, laptops_known); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 278 | } |
Carl-Daniel Hailfinger | 8841d3e | 2010-05-15 15:04:37 +0000 | [diff] [blame] | 279 | #endif |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 280 | |
Uwe Hermann | a6d9c4b | 2010-05-24 15:28:12 +0000 | [diff] [blame] | 281 | static void print_supported_chips_wiki(int cols) |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 282 | { |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 283 | unsigned int lines_per_col; |
Stefan Tauner | 0015549 | 2011-06-26 20:45:35 +0000 | [diff] [blame] | 284 | char *s; |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 285 | char vmax[6]; |
| 286 | char vmin[6]; |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 287 | const struct flashchip *f, *old = NULL; |
| 288 | int i = 0, c = 1, chipcount = 0; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 289 | |
Stefan Tauner | 93f7023 | 2011-07-26 14:33:46 +0000 | [diff] [blame] | 290 | for (f = flashchips; f->name != NULL; f++) { |
Stefan Tauner | 035492a | 2012-02-03 22:32:09 +0000 | [diff] [blame] | 291 | /* Don't count generic entries. */ |
| 292 | if (!strncmp(f->vendor, "Unknown", 7) || |
| 293 | !strncmp(f->vendor, "Programmer", 10) || |
| 294 | !strncmp(f->name, "unknown", 7)) |
Stefan Tauner | 4c5665f | 2012-02-17 20:03:37 +0000 | [diff] [blame] | 295 | continue; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 296 | chipcount++; |
Stefan Tauner | 93f7023 | 2011-07-26 14:33:46 +0000 | [diff] [blame] | 297 | } |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 298 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 299 | /* +1 to force the resulting number of columns to be < cols */ |
| 300 | lines_per_col = chipcount / cols + ((chipcount%cols) > 0 ? 1 : 0); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 301 | |
Stefan Tauner | 1c356bb | 2013-07-11 13:48:52 +0000 | [diff] [blame] | 302 | printf("%s", chip_intro); |
| 303 | printf("\nTotal amount of supported chips: '''%d'''\n\n" |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 304 | "{| border=\"0\" valign=\"top\"\n", chipcount); |
| 305 | |
| 306 | for (f = flashchips; f->name != NULL; f++) { |
Stefan Tauner | 035492a | 2012-02-03 22:32:09 +0000 | [diff] [blame] | 307 | /* Don't print generic entries. */ |
| 308 | if (!strncmp(f->vendor, "Unknown", 7) || |
| 309 | !strncmp(f->vendor, "Programmer", 10) || |
| 310 | !strncmp(f->name, "unknown", 7)) |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 311 | continue; |
| 312 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 313 | if ((i % lines_per_col) == 0) |
| 314 | printf("%s%s", th_start, chip_th); |
| 315 | |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 316 | /* Alternate colors if the vendor changes. */ |
| 317 | if (old != NULL && strcmp(old->vendor, f->vendor)) |
| 318 | c = !c; |
| 319 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 320 | old = f; |
Stefan Tauner | 0015549 | 2011-06-26 20:45:35 +0000 | [diff] [blame] | 321 | s = flashbuses_to_text(f->bustype); |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 322 | sprintf(vmin, "%0.03f", f->voltage.min / (double)1000); |
| 323 | sprintf(vmax, "%0.03f", f->voltage.max / (double)1000); |
Stefan Tauner | f74a7b9 | 2011-09-14 22:09:48 +0000 | [diff] [blame] | 324 | printf("|- bgcolor=\"#%s\"\n| %s || %s || align=\"right\" | %d " |
| 325 | "|| %s || {{%s}} || {{%s}} || {{%s}} || {{%s}}" |
Elyes HAOUAS | ac01baa | 2018-05-28 16:52:21 +0200 | [diff] [blame] | 326 | "|| %s || %s\n", |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 327 | (c == 1) ? "eeeeee" : "dddddd", f->vendor, f->name, |
Stefan Tauner | 0015549 | 2011-06-26 20:45:35 +0000 | [diff] [blame] | 328 | f->total_size, s, |
Stefan Tauner | 618d897 | 2014-05-28 13:02:13 +0000 | [diff] [blame] | 329 | test_state_to_template(f->tested.probe), |
| 330 | test_state_to_template(f->tested.read), |
| 331 | test_state_to_template(f->tested.erase), |
| 332 | test_state_to_template(f->tested.write), |
Stefan Tauner | 6455dff | 2014-05-26 00:36:24 +0000 | [diff] [blame] | 333 | f->voltage.min ? vmin : "?", |
| 334 | f->voltage.max ? vmax : "?"); |
Stefan Tauner | 0015549 | 2011-06-26 20:45:35 +0000 | [diff] [blame] | 335 | free(s); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 336 | |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 337 | if (((i % lines_per_col) + 1) == lines_per_col) |
| 338 | printf("\n|}\n\n"); |
| 339 | i++; |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 340 | } |
Stefan Tauner | 00ec027 | 2012-07-29 02:04:21 +0000 | [diff] [blame] | 341 | /* end inner table if it did not fill the last column fully */ |
| 342 | if (((i % lines_per_col)) > 0) |
| 343 | printf("\n|}\n\n"); |
| 344 | printf("|}\n\n"); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 345 | } |
| 346 | |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 347 | /* Following functions are not needed when no PCI/USB programmers are compiled in, |
| 348 | * but since print_wiki code has no size constraints we include it unconditionally. */ |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 349 | static int count_supported_devs_wiki(const struct dev_entry *devs) |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 350 | { |
| 351 | unsigned int count = 0; |
| 352 | unsigned int i = 0; |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 353 | for (i = 0; devs[i].vendor_id != 0; i++) |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 354 | count++; |
| 355 | return count; |
| 356 | } |
| 357 | |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 358 | static void print_supported_devs_wiki_helper(const struct programmer_entry prog) |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 359 | { |
| 360 | int i = 0; |
| 361 | static int c = 0; |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 362 | const struct dev_entry *devs = prog.devs.dev; |
| 363 | const unsigned int count = count_supported_devs_wiki(devs); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 364 | |
| 365 | /* Alternate colors if the vendor changes. */ |
| 366 | c = !c; |
| 367 | |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 368 | for (i = 0; devs[i].vendor_id != 0; i++) { |
| 369 | printf("|- bgcolor=\"#%s\"\n", (c) ? "eeeeee" : "dddddd"); |
| 370 | if (i == 0) |
| 371 | printf("| rowspan=\"%u\" | %s |", count, prog.name); |
| 372 | printf("| %s || %s || %04x:%04x || {{%s}}\n", devs[i].vendor_name, devs[i].device_name, |
Stefan Tauner | 618d897 | 2014-05-28 13:02:13 +0000 | [diff] [blame] | 373 | devs[i].vendor_id, devs[i].device_id, test_state_to_template(devs[i].status)); |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 374 | } |
| 375 | } |
| 376 | |
| 377 | static void print_supported_devs_wiki() |
| 378 | { |
| 379 | unsigned int pci_count = 0; |
| 380 | unsigned int usb_count = 0; |
| 381 | unsigned int i; |
| 382 | |
| 383 | for (i = 0; i < PROGRAMMER_INVALID; i++) { |
| 384 | const struct programmer_entry prog = programmer_table[i]; |
| 385 | switch (prog.type) { |
| 386 | case USB: |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 387 | usb_count += count_supported_devs_wiki(prog.devs.dev); |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 388 | break; |
| 389 | case PCI: |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 390 | pci_count += count_supported_devs_wiki(prog.devs.dev); |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 391 | break; |
| 392 | case OTHER: |
| 393 | default: |
| 394 | break; |
| 395 | } |
| 396 | } |
| 397 | |
| 398 | printf("\n== PCI Devices ==\n\n" |
| 399 | "Total amount of supported PCI devices flashrom can use as a programmer: '''%d'''\n\n" |
| 400 | "{%s%s", pci_count, th_start, programmer_th); |
| 401 | |
| 402 | for (i = 0; i < PROGRAMMER_INVALID; i++) { |
| 403 | const struct programmer_entry prog = programmer_table[i]; |
| 404 | if (prog.type == PCI) { |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 405 | print_supported_devs_wiki_helper(prog); |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 406 | } |
| 407 | } |
| 408 | printf("\n|}\n\n|}\n"); |
| 409 | |
| 410 | printf("\n== USB Devices ==\n\n" |
| 411 | "Total amount of supported USB devices flashrom can use as a programmer: '''%d'''\n\n" |
| 412 | "{%s%s", usb_count, th_start, programmer_th); |
| 413 | |
| 414 | for (i = 0; i < PROGRAMMER_INVALID; i++) { |
| 415 | const struct programmer_entry prog = programmer_table[i]; |
| 416 | if (prog.type == USB) { |
Stefan Tauner | 4b24a2d | 2012-12-27 18:40:36 +0000 | [diff] [blame] | 417 | print_supported_devs_wiki_helper(prog); |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 418 | } |
| 419 | } |
| 420 | printf("\n|}\n\n|}\n"); |
Stefan Tauner | 2c42119 | 2012-12-27 18:40:40 +0000 | [diff] [blame] | 421 | |
| 422 | printf("\n== Other programmers ==\n\n" |
| 423 | "{%s", th_start); |
| 424 | printf("! align=\"left\" | Programmer\n" |
| 425 | "! align=\"left\" | Note\n\n"); |
| 426 | |
| 427 | for (i = 0; i < PROGRAMMER_INVALID; i++) { |
| 428 | static int c = 0; |
| 429 | const struct programmer_entry prog = programmer_table[i]; |
| 430 | if (prog.type == OTHER && prog.devs.note != NULL) { |
| 431 | c = !c; |
| 432 | printf("|- bgcolor=\"#%s\"\n", (c) ? "eeeeee" : "dddddd"); |
| 433 | printf("| %s || %s", prog.name, prog.devs.note); |
| 434 | } |
| 435 | } |
| 436 | printf("\n|}\n\n|}\n"); |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 437 | } |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 438 | |
Carl-Daniel Hailfinger | f529205 | 2009-11-17 09:57:34 +0000 | [diff] [blame] | 439 | void print_supported_wiki(void) |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 440 | { |
| 441 | time_t t = time(NULL); |
Stefan Tauner | 2a10e70 | 2014-06-02 00:46:02 +0000 | [diff] [blame] | 442 | char buf[sizeof("1986-02-28T12:37:42Z")]; |
| 443 | strftime(buf, sizeof(buf), "%Y-%m-%dT%H:%M:%SZ", gmtime(&t)); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 444 | |
Stefan Tauner | 2a10e70 | 2014-06-02 00:46:02 +0000 | [diff] [blame] | 445 | printf(wiki_header, buf, flashrom_version); |
Uwe Hermann | a6d9c4b | 2010-05-24 15:28:12 +0000 | [diff] [blame] | 446 | print_supported_chips_wiki(2); |
Carl-Daniel Hailfinger | 9e3a6c4 | 2010-10-08 12:40:09 +0000 | [diff] [blame] | 447 | #if CONFIG_INTERNAL == 1 |
Uwe Hermann | a6d9c4b | 2010-05-24 15:28:12 +0000 | [diff] [blame] | 448 | print_supported_chipsets_wiki(3); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 449 | print_supported_boards_wiki(); |
Carl-Daniel Hailfinger | 8841d3e | 2010-05-15 15:04:37 +0000 | [diff] [blame] | 450 | #endif |
Stefan Tauner | af358d6 | 2012-12-27 18:40:26 +0000 | [diff] [blame] | 451 | print_supported_devs_wiki(); |
Carl-Daniel Hailfinger | 9c8476b | 2009-09-16 12:19:03 +0000 | [diff] [blame] | 452 | } |
| 453 | |