blob: 5d85758d72ade95c8bb7db55ca15e25b343597bc [file] [log] [blame]
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +00001/*
2 * This file is part of the flashrom project.
3 *
4 * Copyright (C) 2009 Carl-Daniel Hailfinger
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.
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +000015 */
16
Carl-Daniel Hailfinger1c6d2ff2012-08-27 00:44:42 +000017#include <strings.h>
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +000018#include <string.h>
Felix Singerb8db74a2022-08-19 00:19:26 +020019#include <stdbool.h>
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +000020#include <stdlib.h>
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +000021#include "flash.h"
Carl-Daniel Hailfinger5b997c32010-07-27 22:41:39 +000022#include "programmer.h"
Thomas Heijligen74b4aa02021-12-14 17:52:30 +010023#include "hwaccess_physmap.h"
Thomas Heijligend96c97c2021-11-02 21:03:00 +010024#include "platform/pci.h"
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +000025
Peter Marheinedf3672d2022-01-19 17:11:09 +110026#if defined(__i386__) || defined(__x86_64__)
27#include "hwaccess_x86_io.h"
28#endif
29
Felix Singerb8db74a2022-08-19 00:19:26 +020030bool force_boardenable = false;
31bool force_boardmismatch = false;
Carl-Daniel Hailfinger14e100c2009-12-22 23:42:04 +000032
Thomas Heijligen4b918a12021-09-26 13:42:39 +020033#if defined(__i386__) || defined(__x86_64__)
Carl-Daniel Hailfinger14e100c2009-12-22 23:42:04 +000034void probe_superio(void)
35{
Carl-Daniel Hailfingerf5e62cb2012-05-06 22:48:01 +000036 probe_superio_winbond();
37 /* ITE probe causes SMSC LPC47N217 to power off the serial UART.
38 * Always probe for SMSC first, and if a SMSC Super I/O is detected
39 * at a given I/O port, do _not_ probe that port with the ITE probe.
40 * This means SMSC probing must be done before ITE probing.
41 */
42 //probe_superio_smsc();
Carl-Daniel Hailfingerbfecef62011-04-27 14:34:08 +000043 probe_superio_ite();
Carl-Daniel Hailfinger14e100c2009-12-22 23:42:04 +000044}
Carl-Daniel Hailfingerbfecef62011-04-27 14:34:08 +000045
46int superio_count = 0;
47#define SUPERIO_MAX_COUNT 3
48
49struct superio superios[SUPERIO_MAX_COUNT];
50
51int register_superio(struct superio s)
52{
53 if (superio_count == SUPERIO_MAX_COUNT)
54 return 1;
55 superios[superio_count++] = s;
56 return 0;
57}
58
Carl-Daniel Hailfingercceafa22010-05-26 01:45:41 +000059#endif
Carl-Daniel Hailfinger14e100c2009-12-22 23:42:04 +000060
Carl-Daniel Hailfingercceafa22010-05-26 01:45:41 +000061int is_laptop = 0;
Felix Singerd1ab7d22022-08-19 03:03:47 +020062bool laptop_ok = false;
Michael Karcher8c1df282010-02-26 09:51:20 +000063
Carl-Daniel Hailfinger8a3c60c2011-12-18 15:01:24 +000064static void internal_chip_writeb(const struct flashctx *flash, uint8_t val,
65 chipaddr addr);
66static void internal_chip_writew(const struct flashctx *flash, uint16_t val,
67 chipaddr addr);
68static void internal_chip_writel(const struct flashctx *flash, uint32_t val,
69 chipaddr addr);
70static uint8_t internal_chip_readb(const struct flashctx *flash,
71 const chipaddr addr);
72static uint16_t internal_chip_readw(const struct flashctx *flash,
73 const chipaddr addr);
74static uint32_t internal_chip_readl(const struct flashctx *flash,
75 const chipaddr addr);
76static void internal_chip_readn(const struct flashctx *flash, uint8_t *buf,
77 const chipaddr addr, size_t len);
Carl-Daniel Hailfingera5bcbce2014-07-19 22:03:29 +000078static const struct par_master par_master_internal = {
Thomas Heijligen43040f22022-06-23 14:38:35 +020079 .chip_readb = internal_chip_readb,
80 .chip_readw = internal_chip_readw,
81 .chip_readl = internal_chip_readl,
82 .chip_readn = internal_chip_readn,
83 .chip_writeb = internal_chip_writeb,
84 .chip_writew = internal_chip_writew,
85 .chip_writel = internal_chip_writel,
86 .chip_writen = fallback_chip_writen,
Nico Huber0e76d992023-01-12 20:22:55 +010087 .map_flash = physmap,
88 .unmap_flash = physunmap,
Carl-Daniel Hailfingereaacd2d2011-11-09 23:40:00 +000089};
90
91enum chipbustype internal_buses_supported = BUS_NONE;
92
Felix Singerb8db74a2022-08-19 00:19:26 +020093static int get_params(bool *boardenable, bool *boardmismatch,
94 bool *force_laptop, bool *not_a_laptop,
Edward O'Callaghand91ee2c2022-02-03 11:55:13 +110095 char **board_vendor, char **board_model)
Uwe Hermanna0869322009-05-14 20:41:57 +000096{
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +000097 char *arg;
Uwe Hermanna0869322009-05-14 20:41:57 +000098
Edward O'Callaghand91ee2c2022-02-03 11:55:13 +110099 /* default values. */
Felix Singerb8db74a2022-08-19 00:19:26 +0200100 *force_laptop = false;
101 *not_a_laptop = false;
Edward O'Callaghand91ee2c2022-02-03 11:55:13 +1100102 *board_vendor = NULL;
103 *board_model = NULL;
104
Carl-Daniel Hailfinger2b6dcb32010-07-08 10:13:37 +0000105 arg = extract_programmer_param("boardenable");
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000106 if (arg && !strcmp(arg,"force")) {
Felix Singerb8db74a2022-08-19 00:19:26 +0200107 *boardenable = true;
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000108 } else if (arg && !strlen(arg)) {
109 msg_perr("Missing argument for boardenable.\n");
Carl-Daniel Hailfinger744132a2010-07-06 09:55:48 +0000110 free(arg);
111 return 1;
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000112 } else if (arg) {
113 msg_perr("Unknown argument for boardenable: %s\n", arg);
Carl-Daniel Hailfinger744132a2010-07-06 09:55:48 +0000114 free(arg);
115 return 1;
Michael Karcher0bdc0922010-02-28 01:33:48 +0000116 }
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000117 free(arg);
Michael Karcher0bdc0922010-02-28 01:33:48 +0000118
Carl-Daniel Hailfinger2b6dcb32010-07-08 10:13:37 +0000119 arg = extract_programmer_param("boardmismatch");
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000120 if (arg && !strcmp(arg,"force")) {
Felix Singerb8db74a2022-08-19 00:19:26 +0200121 *boardmismatch = true;
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000122 } else if (arg && !strlen(arg)) {
123 msg_perr("Missing argument for boardmismatch.\n");
Carl-Daniel Hailfinger744132a2010-07-06 09:55:48 +0000124 free(arg);
125 return 1;
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000126 } else if (arg) {
127 msg_perr("Unknown argument for boardmismatch: %s\n", arg);
Carl-Daniel Hailfinger744132a2010-07-06 09:55:48 +0000128 free(arg);
129 return 1;
Michael Karcher0bdc0922010-02-28 01:33:48 +0000130 }
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000131 free(arg);
132
Carl-Daniel Hailfinger2b6dcb32010-07-08 10:13:37 +0000133 arg = extract_programmer_param("laptop");
Stefan Taunera28087f2011-09-13 23:14:25 +0000134 if (arg && !strcmp(arg, "force_I_want_a_brick"))
Felix Singerb8db74a2022-08-19 00:19:26 +0200135 *force_laptop = true;
Stefan Taunera28087f2011-09-13 23:14:25 +0000136 else if (arg && !strcmp(arg, "this_is_not_a_laptop"))
Felix Singerb8db74a2022-08-19 00:19:26 +0200137 *not_a_laptop = true;
Stefan Taunera28087f2011-09-13 23:14:25 +0000138 else if (arg && !strlen(arg)) {
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000139 msg_perr("Missing argument for laptop.\n");
Carl-Daniel Hailfinger744132a2010-07-06 09:55:48 +0000140 free(arg);
141 return 1;
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000142 } else if (arg) {
143 msg_perr("Unknown argument for laptop: %s\n", arg);
Carl-Daniel Hailfinger744132a2010-07-06 09:55:48 +0000144 free(arg);
145 return 1;
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000146 }
147 free(arg);
148
Carl-Daniel Hailfinger2d927fb2012-01-04 00:48:27 +0000149 arg = extract_programmer_param("mainboard");
150 if (arg && strlen(arg)) {
Edward O'Callaghand91ee2c2022-02-03 11:55:13 +1100151 if (board_parse_parameter(arg, board_vendor, board_model)) {
Stefan Taunerb4e06bd2012-08-20 00:24:22 +0000152 free(arg);
153 return 1;
154 }
Carl-Daniel Hailfinger2d927fb2012-01-04 00:48:27 +0000155 } else if (arg && !strlen(arg)) {
156 msg_perr("Missing argument for mainboard.\n");
157 free(arg);
158 return 1;
159 }
160 free(arg);
161
Edward O'Callaghand91ee2c2022-02-03 11:55:13 +1100162 return 0;
163}
164
165static int internal_init(void)
166{
167 int ret = 0;
Felix Singerb8db74a2022-08-19 00:19:26 +0200168 bool force_laptop;
169 bool not_a_laptop;
Edward O'Callaghand91ee2c2022-02-03 11:55:13 +1100170 char *board_vendor;
171 char *board_model;
172#if defined(__i386__) || defined(__x86_64__)
173 const char *cb_vendor = NULL;
174 const char *cb_model = NULL;
175#endif
176
177 ret = get_params(&force_boardenable, &force_boardmismatch,
178 &force_laptop, &not_a_laptop,
179 &board_vendor, &board_model);
180 if (ret)
181 return ret;
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000182
Edward O'Callaghanbd275812022-11-28 11:20:44 +1100183 /* Unconditionally reset global state from previous operation. */
184 laptop_ok = false;
185
Michael Karcherb9dbe482011-05-11 17:07:07 +0000186 /* Default to Parallel/LPC/FWH flash devices. If a known host controller
Carl-Daniel Hailfingereaacd2d2011-11-09 23:40:00 +0000187 * is found, the host controller init routine sets the
188 * internal_buses_supported bitfield.
Michael Karcherb9dbe482011-05-11 17:07:07 +0000189 */
Carl-Daniel Hailfingereaacd2d2011-11-09 23:40:00 +0000190 internal_buses_supported = BUS_NONSPI;
Michael Karcherb9dbe482011-05-11 17:07:07 +0000191
Jacob Garber1c091d12019-08-12 11:14:14 -0600192 if (try_mtd() == 0) {
193 ret = 0;
194 goto internal_init_exit;
195 }
David Hendricksf9a30552015-05-23 20:30:30 -0700196
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000197 /* Initialize PCI access for flash enables */
Jacob Garber1c091d12019-08-12 11:14:14 -0600198 if (pci_init_common() != 0) {
199 ret = 1;
200 goto internal_init_exit;
201 }
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000202
Carl-Daniel Hailfingerb5b161b2010-06-04 19:05:39 +0000203 if (processor_flash_enable()) {
204 msg_perr("Processor detection/init failed.\n"
205 "Aborting.\n");
Jacob Garber1c091d12019-08-12 11:14:14 -0600206 ret = 1;
207 goto internal_init_exit;
Carl-Daniel Hailfingerb5b161b2010-06-04 19:05:39 +0000208 }
209
Thomas Heijligen4b918a12021-09-26 13:42:39 +0200210#if defined(__i386__) || defined(__x86_64__)
Peter Marheinedf3672d2022-01-19 17:11:09 +1100211 if (rget_io_perms()) {
212 ret = 1;
213 goto internal_init_exit;
214 }
215
Stefan Taunerfa9fa712012-09-24 21:29:29 +0000216 if ((cb_parse_table(&cb_vendor, &cb_model) == 0) && (board_vendor != NULL) && (board_model != NULL)) {
217 if (strcasecmp(board_vendor, cb_vendor) || strcasecmp(board_model, cb_model)) {
Stefan Taunerc6fa32d2013-01-04 22:54:07 +0000218 msg_pwarn("Warning: The mainboard IDs set by -p internal:mainboard (%s:%s) do not\n"
Stefan Taunerb4e06bd2012-08-20 00:24:22 +0000219 " match the current coreboot IDs of the mainboard (%s:%s).\n",
220 board_vendor, board_model, cb_vendor, cb_model);
Jacob Garber1c091d12019-08-12 11:14:14 -0600221 if (!force_boardmismatch) {
222 ret = 1;
223 goto internal_init_exit;
224 }
Stefan Taunerb4e06bd2012-08-20 00:24:22 +0000225 msg_pinfo("Continuing anyway.\n");
226 }
227 }
Carl-Daniel Hailfingercceafa22010-05-26 01:45:41 +0000228
Nico Huber2e50cdc2018-09-23 20:20:26 +0200229 is_laptop = 2; /* Assume that we don't know by default. */
230
Michael Karcher6701ee82010-01-20 14:14:11 +0000231 dmi_init();
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000232
Carl-Daniel Hailfinger580d29a2011-05-05 07:12:40 +0000233 /* In case Super I/O probing would cause pretty explosions. */
234 board_handle_before_superio();
235
Uwe Hermann43959702010-03-13 17:28:29 +0000236 /* Probe for the Super I/O chip and fill global struct superio. */
Carl-Daniel Hailfinger14e100c2009-12-22 23:42:04 +0000237 probe_superio();
Carl-Daniel Hailfingerb5b161b2010-06-04 19:05:39 +0000238#else
239 /* FIXME: Enable cbtable searching on all non-x86 platforms supported
240 * by coreboot.
241 * FIXME: Find a replacement for DMI on non-x86.
242 * FIXME: Enable Super I/O probing once port I/O is possible.
243 */
Carl-Daniel Hailfingercceafa22010-05-26 01:45:41 +0000244#endif
Carl-Daniel Hailfinger14e100c2009-12-22 23:42:04 +0000245
Carl-Daniel Hailfinger580d29a2011-05-05 07:12:40 +0000246 /* Check laptop whitelist. */
247 board_handle_before_laptop();
248
Nico Huber2e50cdc2018-09-23 20:20:26 +0200249 /*
250 * Disable all internal buses by default if we are not sure
251 * this isn't a laptop. Board-enables may override this,
252 * non-legacy buses (SPI and opaque atm) are probed anyway.
253 */
254 if (is_laptop && !(laptop_ok || force_laptop || (not_a_laptop && is_laptop == 2)))
255 internal_buses_supported = BUS_NONE;
Michael Karcher8c1df282010-02-26 09:51:20 +0000256
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000257 /* try to enable it. Failure IS an option, since not all motherboards
258 * really need this to be done, etc., etc.
259 */
260 ret = chipset_flash_enable();
261 if (ret == -2) {
Carl-Daniel Hailfinger27023762010-04-28 15:22:14 +0000262 msg_perr("WARNING: No chipset found. Flash detection "
263 "will most likely fail.\n");
Jacob Garber1c091d12019-08-12 11:14:14 -0600264 } else if (ret == ERROR_FATAL) {
265 goto internal_init_exit;
266 }
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000267
Thomas Heijligen4b918a12021-09-26 13:42:39 +0200268#if defined(__i386__) || defined(__x86_64__)
Vadim Girlin4dd0f902013-08-24 12:18:17 +0000269 /* Probe unconditionally for ITE Super I/O chips. This enables LPC->SPI translation on IT87* and
270 * parallel writes on IT8705F. Also, this handles the manual chip select for Gigabyte's DualBIOS. */
Carl-Daniel Hailfinger76d4b372010-07-10 16:56:32 +0000271 init_superio_ite();
Carl-Daniel Hailfinger01f3ef42010-03-25 02:50:40 +0000272
Stefan Taunerfa9fa712012-09-24 21:29:29 +0000273 if (board_flash_enable(board_vendor, board_model, cb_vendor, cb_model)) {
Stefan Taunerb4e06bd2012-08-20 00:24:22 +0000274 msg_perr("Aborting to be safe.\n");
Jacob Garber1c091d12019-08-12 11:14:14 -0600275 ret = 1;
276 goto internal_init_exit;
Stefan Taunerb4e06bd2012-08-20 00:24:22 +0000277 }
Stefan Taunerb0eee9b2015-01-10 09:32:50 +0000278#endif
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000279
Nico Huber2e50cdc2018-09-23 20:20:26 +0200280 if (internal_buses_supported & BUS_NONSPI)
Anastasia Klimchukb91a2032021-05-21 09:40:58 +1000281 register_par_master(&par_master_internal, internal_buses_supported, NULL);
Nico Huber2e50cdc2018-09-23 20:20:26 +0200282
283 /* Report if a non-whitelisted laptop is detected that likely uses a legacy bus. */
284 if (is_laptop && !laptop_ok) {
285 msg_pinfo("========================================================================\n");
286 if (is_laptop == 1) {
Nico Huberc3b02dc2023-08-12 01:13:45 +0200287 msg_pinfo("You seem to be running flashprog on an unknown laptop. Some\n"
Nico Huber2e50cdc2018-09-23 20:20:26 +0200288 "internal buses have been disabled for safety reasons.\n\n");
289 } else {
Nico Huberc3b02dc2023-08-12 01:13:45 +0200290 msg_pinfo("You may be running flashprog on an unknown laptop. We could not\n"
Nico Huber2e50cdc2018-09-23 20:20:26 +0200291 "detect this for sure because your vendor has not set up the SMBIOS\n"
292 "tables correctly. Some internal buses have been disabled for\n"
293 "safety reasons. You can enforce using all buses by adding\n"
294 " -p internal:laptop=this_is_not_a_laptop\n"
295 "to the command line, but please read the following warning if you\n"
296 "are not sure.\n\n");
297 }
298 msg_perr("Laptops, notebooks and netbooks are difficult to support and we\n"
299 "recommend to use the vendor flashing utility. The embedded controller\n"
300 "(EC) in these machines often interacts badly with flashing.\n"
Nico Huberc3b02dc2023-08-12 01:13:45 +0200301 "See the manpage and https://flashprog.org/Laptops for details.\n\n"
Nico Huber2e50cdc2018-09-23 20:20:26 +0200302 "If flash is shared with the EC, erase is guaranteed to brick your laptop\n"
303 "and write may brick your laptop.\n"
304 "Read and probe may irritate your EC and cause fan failure, backlight\n"
305 "failure and sudden poweroff.\n"
306 "You have been warned.\n"
307 "========================================================================\n");
308 }
309
Jacob Garber1c091d12019-08-12 11:14:14 -0600310 ret = 0;
311
312internal_init_exit:
313 free(board_vendor);
314 free(board_model);
315
316 return ret;
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000317}
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000318
Carl-Daniel Hailfinger8a3c60c2011-12-18 15:01:24 +0000319static void internal_chip_writeb(const struct flashctx *flash, uint8_t val,
320 chipaddr addr)
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000321{
Carl-Daniel Hailfinger78185dc2009-05-17 15:49:24 +0000322 mmio_writeb(val, (void *) addr);
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000323}
324
Carl-Daniel Hailfinger8a3c60c2011-12-18 15:01:24 +0000325static void internal_chip_writew(const struct flashctx *flash, uint16_t val,
326 chipaddr addr)
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000327{
Carl-Daniel Hailfinger78185dc2009-05-17 15:49:24 +0000328 mmio_writew(val, (void *) addr);
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000329}
330
Carl-Daniel Hailfinger8a3c60c2011-12-18 15:01:24 +0000331static void internal_chip_writel(const struct flashctx *flash, uint32_t val,
332 chipaddr addr)
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000333{
Carl-Daniel Hailfinger78185dc2009-05-17 15:49:24 +0000334 mmio_writel(val, (void *) addr);
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000335}
336
Carl-Daniel Hailfinger8a3c60c2011-12-18 15:01:24 +0000337static uint8_t internal_chip_readb(const struct flashctx *flash,
338 const chipaddr addr)
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000339{
Carl-Daniel Hailfinger78185dc2009-05-17 15:49:24 +0000340 return mmio_readb((void *) addr);
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000341}
342
Carl-Daniel Hailfinger8a3c60c2011-12-18 15:01:24 +0000343static uint16_t internal_chip_readw(const struct flashctx *flash,
344 const chipaddr addr)
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000345{
Carl-Daniel Hailfinger78185dc2009-05-17 15:49:24 +0000346 return mmio_readw((void *) addr);
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000347}
348
Carl-Daniel Hailfinger8a3c60c2011-12-18 15:01:24 +0000349static uint32_t internal_chip_readl(const struct flashctx *flash,
350 const chipaddr addr)
Carl-Daniel Hailfinger702218d2009-05-08 17:43:22 +0000351{
Carl-Daniel Hailfinger78185dc2009-05-17 15:49:24 +0000352 return mmio_readl((void *) addr);
353}
354
Carl-Daniel Hailfinger8a3c60c2011-12-18 15:01:24 +0000355static void internal_chip_readn(const struct flashctx *flash, uint8_t *buf,
356 const chipaddr addr, size_t len)
Carl-Daniel Hailfinger0bd2a2b2009-06-05 18:32:07 +0000357{
Carl-Daniel Hailfingerccd71c22012-03-01 22:38:27 +0000358 mmio_readn((void *)addr, buf, len);
Carl-Daniel Hailfinger0bd2a2b2009-06-05 18:32:07 +0000359 return;
360}
Thomas Heijligencc853d82021-05-04 15:32:17 +0200361
362const struct programmer_entry programmer_internal = {
363 .name = "internal",
364 .type = OTHER,
365 .devs.note = NULL,
366 .init = internal_init,
Thomas Heijligencc853d82021-05-04 15:32:17 +0200367};