xref: /freebsd/usr.sbin/bluetooth/iwmbtfw/iwmbt_hw.c (revision 13ec1e3155c7e9bf037b12af186351b7fa9b9450)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2019 Vladimir Kondratyev <wulf@FreeBSD.org>
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25  * SUCH DAMAGE.
26  *
27  * $FreeBSD$
28  */
29 
30 #include <sys/param.h>
31 #include <sys/endian.h>
32 #include <sys/stat.h>
33 
34 #include <err.h>
35 #include <errno.h>
36 #include <stddef.h>
37 #include <stdio.h>
38 #include <stdlib.h>
39 #include <string.h>
40 #include <unistd.h>
41 
42 #include <libusb.h>
43 
44 #include "iwmbt_fw.h"
45 #include "iwmbt_hw.h"
46 #include "iwmbt_dbg.h"
47 
48 #define	XMIN(x, y)	((x) < (y) ? (x) : (y))
49 
50 static int
51 iwmbt_send_fragment(struct libusb_device_handle *hdl,
52     uint8_t fragment_type, const void *data, uint8_t len, int timeout)
53 {
54 	int ret, transferred;
55 	uint8_t buf[IWMBT_HCI_MAX_CMD_SIZE];
56 	struct iwmbt_hci_cmd *cmd = (struct iwmbt_hci_cmd *) buf;
57 
58 	memset(buf, 0, sizeof(buf));
59 	cmd->opcode = htole16(0xfc09),
60 	cmd->length = len + 1,
61 	cmd->data[0] = fragment_type;
62 	memcpy(cmd->data + 1, data, len);
63 
64 	ret = libusb_bulk_transfer(hdl,
65 	    IWMBT_BULK_OUT_ENDPOINT_ADDR,
66 	    (uint8_t *)cmd,
67 	    IWMBT_HCI_CMD_SIZE(cmd),
68 	    &transferred,
69 	    timeout);
70 
71 	if (ret < 0 || transferred != (int)IWMBT_HCI_CMD_SIZE(cmd)) {
72 		iwmbt_err("libusb_bulk_transfer() failed: err=%s, size=%zu",
73 		    libusb_strerror(ret),
74 		    IWMBT_HCI_CMD_SIZE(cmd));
75 		return (-1);
76 	}
77 
78 	ret = libusb_bulk_transfer(hdl,
79 	    IWMBT_BULK_IN_ENDPOINT_ADDR,
80 	    buf,
81 	    sizeof(buf),
82 	    &transferred,
83 	    timeout);
84 
85 	if (ret < 0) {
86 		iwmbt_err("libusb_bulk_transfer() failed: err=%s",
87 		    libusb_strerror(ret));
88 		return (-1);
89 	}
90 
91 	return (0);
92 }
93 
94 static int
95 iwmbt_hci_command(struct libusb_device_handle *hdl, struct iwmbt_hci_cmd *cmd,
96     void *event, int size, int *transferred, int timeout)
97 {
98 	int ret;
99 
100 	ret = libusb_control_transfer(hdl,
101 	    LIBUSB_REQUEST_TYPE_CLASS | LIBUSB_RECIPIENT_DEVICE,
102 	    0,
103 	    0,
104 	    0,
105 	    (uint8_t *)cmd,
106 	    IWMBT_HCI_CMD_SIZE(cmd),
107 	    timeout);
108 
109 	if (ret < 0) {
110 		iwmbt_err("libusb_control_transfer() failed: err=%s",
111 		    libusb_strerror(ret));
112 		return (ret);
113 	}
114 
115 	ret = libusb_interrupt_transfer(hdl,
116 	    IWMBT_INTERRUPT_ENDPOINT_ADDR,
117 	    event,
118 	    size,
119 	    transferred,
120 	    timeout);
121 
122 	if (ret < 0)
123 		iwmbt_err("libusb_interrupt_transfer() failed: err=%s",
124 		    libusb_strerror(ret));
125 
126 	return (ret);
127 }
128 
129 int
130 iwmbt_patch_fwfile(struct libusb_device_handle *hdl,
131     const struct iwmbt_firmware *fw)
132 {
133 	int ret, transferred;
134 	struct iwmbt_firmware fw_job = *fw;
135 	uint16_t cmd_opcode;
136 	uint8_t cmd_length;
137 	struct iwmbt_hci_cmd *cmd_buf;
138 	uint8_t evt_code;
139 	uint8_t evt_length;
140 	uint8_t evt_buf[IWMBT_HCI_MAX_EVENT_SIZE];
141 	int activate_patch = 0;
142 
143 	while (fw_job.len > 0) {
144 		if (fw_job.len < 4) {
145 			iwmbt_err("Invalid firmware, unexpected EOF in HCI "
146 			    "command header. Remains=%d", fw_job.len);
147 			return (-1);
148 		}
149 
150 		if (fw_job.buf[0] != 0x01) {
151 			iwmbt_err("Invalid firmware, expected HCI command (%d)",
152 					fw_job.buf[0]);
153 			return (-1);
154 		}
155 
156 		/* Advance by one. */
157 		fw_job.buf++;
158 		fw_job.len--;
159 
160 		/* Load in the HCI command to perform. */
161 		cmd_opcode = le16dec(fw_job.buf);
162 		cmd_length = fw_job.buf[2];
163 		cmd_buf = (struct iwmbt_hci_cmd *)fw_job.buf;
164 
165 		iwmbt_debug("opcode=%04x, len=%02x", cmd_opcode, cmd_length);
166 
167 		/*
168 		 * If there is a command that loads a patch in the
169 		 * firmware file, then activate the patch upon success,
170 		 * otherwise just disable the manufacturer mode.
171 		 */
172 		if (cmd_opcode == 0xfc8e)
173 			activate_patch = 1;
174 
175 		/* Advance by three. */
176 		fw_job.buf += 3;
177 		fw_job.len -= 3;
178 
179 		if (fw_job.len < cmd_length) {
180 			iwmbt_err("Invalid firmware, unexpected EOF in HCI "
181 			    "command data. len=%d, remains=%d",
182 			    cmd_length, fw_job.len);
183 			return (-1);
184 		}
185 
186 		/* Advance by data length. */
187 		fw_job.buf += cmd_length;
188 		fw_job.len -= cmd_length;
189 
190 		ret = libusb_control_transfer(hdl,
191 		    LIBUSB_REQUEST_TYPE_CLASS | LIBUSB_RECIPIENT_DEVICE,
192 		    0,
193 		    0,
194 		    0,
195 		    (uint8_t *)cmd_buf,
196 		    IWMBT_HCI_CMD_SIZE(cmd_buf),
197 		    IWMBT_HCI_CMD_TIMEOUT);
198 
199 		if (ret < 0) {
200 			iwmbt_err("libusb_control_transfer() failed: err=%s",
201 			    libusb_strerror(ret));
202 			return (-1);
203 		}
204 
205 		/*
206 		 * Every command has its associated event: data must match
207 		 * what is recorded in the firmware file. Perform that check
208 		 * now.
209 		 */
210 
211 		while (fw_job.len > 0 && fw_job.buf[0] == 0x02) {
212 			/* Is this the end of the file? */
213 			if (fw_job.len < 3) {
214 				iwmbt_err("Invalid firmware, unexpected EOF in"
215 				    "event header. remains=%d", fw_job.len);
216 				return (-1);
217 			}
218 
219 			/* Advance by one. */
220 			fw_job.buf++;
221 			fw_job.len--;
222 
223 			/* Load in the HCI event. */
224 			evt_code = fw_job.buf[0];
225 			evt_length = fw_job.buf[1];
226 
227 			/* Advance by two. */
228 			fw_job.buf += 2;
229 			fw_job.len -= 2;
230 
231 			/* Prepare HCI event buffer. */
232 			memset(evt_buf, 0, IWMBT_HCI_MAX_EVENT_SIZE);
233 
234 			iwmbt_debug("event=%04x, len=%02x",
235 					evt_code, evt_length);
236 
237 			if (fw_job.len < evt_length) {
238 				iwmbt_err("Invalid firmware, unexpected EOF in"
239 				    " event data. len=%d, remains=%d",
240 				    evt_length, fw_job.len);
241 				return (-1);
242 			}
243 
244 			ret = libusb_interrupt_transfer(hdl,
245 			    IWMBT_INTERRUPT_ENDPOINT_ADDR,
246 			    evt_buf,
247 			    IWMBT_HCI_MAX_EVENT_SIZE,
248 			    &transferred,
249 			    IWMBT_HCI_CMD_TIMEOUT);
250 
251 			if (ret < 0) {
252 				iwmbt_err("libusb_interrupt_transfer() failed:"
253 				    " err=%s", libusb_strerror(ret));
254 				return (-1);
255 			}
256 
257 			if ((int)evt_length + 2 != transferred ||
258 			    memcmp(evt_buf + 2, fw_job.buf, evt_length) != 0) {
259 				iwmbt_err("event does not match firmware");
260 				return (-1);
261 			}
262 
263 			/* Advance by data length. */
264 			fw_job.buf += evt_length;
265 			fw_job.len -= evt_length;
266 		}
267 	}
268 
269 	return (activate_patch);
270 }
271 
272 int
273 iwmbt_load_fwfile(struct libusb_device_handle *hdl,
274     const struct iwmbt_firmware *fw, uint32_t *boot_param)
275 {
276 	int ready = 0, sent = 0;
277 	int ret, transferred;
278 	struct iwmbt_hci_cmd *cmd;
279 	struct iwmbt_hci_event *event;
280 	uint8_t buf[IWMBT_HCI_MAX_EVENT_SIZE];
281 
282 #define	IWMBT_SEND_FRAGMENT(fragment_type, size, msg)	do {		\
283 	iwmbt_debug("transferring %d bytes, offset %d", size, sent);	\
284 									\
285 	ret = iwmbt_send_fragment(hdl,					\
286 	    fragment_type,						\
287 	    fw->buf + sent,						\
288 	    XMIN(size, fw->len - sent),					\
289 	    IWMBT_HCI_CMD_TIMEOUT);					\
290 									\
291 	if (ret < 0) {							\
292 		iwmbt_debug("Failed to send "msg": code=%d", ret);	\
293 		return (-1);						\
294 	}								\
295 	sent += size;							\
296 } while (0)
297 
298 	if (fw->len < 644) {
299 		iwmbt_err("Invalid size of firmware file (%d)", fw->len);
300 		return (-1);
301 	}
302 
303 	iwmbt_debug("file=%s, size=%d", fw->fwname, fw->len);
304 
305 	IWMBT_SEND_FRAGMENT(0x00, 0x80, "CCS segment");
306 	IWMBT_SEND_FRAGMENT(0x03, 0x80, "public key / part 1");
307 	IWMBT_SEND_FRAGMENT(0x03, 0x80, "public key / part 2");
308 
309 	/* skip 4 bytes */
310 	sent += 4;
311 
312 	IWMBT_SEND_FRAGMENT(0x02, 0x80, "signature / part 1");
313 	IWMBT_SEND_FRAGMENT(0x02, 0x80, "signature / part 2");
314 
315 	/*
316 	 * Send firmware chunks. Chunk len must be 4 byte aligned.
317 	 * multiple commands can be combined
318 	 */
319 	while (fw->len - sent - ready >= (int) sizeof(struct iwmbt_hci_cmd)) {
320 		cmd = (struct iwmbt_hci_cmd *)(fw->buf + sent + ready);
321 		/* Parse firmware for Intel Reset HCI command parameter */
322 		if (cmd->opcode == htole16(0xfc0e)) {
323 			*boot_param = le32dec(cmd->data);
324 			iwmbt_debug("boot_param=0x%08x", *boot_param);
325 		}
326 		ready += IWMBT_HCI_CMD_SIZE(cmd);
327 		while (ready >= 0xFC) {
328 			IWMBT_SEND_FRAGMENT(0x01, 0xFC, "firmware chunk");
329 			ready -= 0xFC;
330 		}
331 		if (ready > 0 && ready % 4 == 0) {
332 			IWMBT_SEND_FRAGMENT(0x01, ready, "firmware chunk");
333 			ready = 0;
334 		}
335 	}
336 
337 	/* Wait for firmware download completion event */
338 	ret = libusb_interrupt_transfer(hdl,
339 	    IWMBT_INTERRUPT_ENDPOINT_ADDR,
340 	    buf,
341 	    sizeof(buf),
342 	    &transferred,
343 	    IWMBT_LOADCMPL_TIMEOUT);
344 
345 	if (ret < 0 || transferred < (int)sizeof(struct iwmbt_hci_event) + 1) {
346 		iwmbt_err("libusb_interrupt_transfer() failed: "
347 		    "err=%s, size=%d",
348 		    libusb_strerror(ret),
349 		    transferred);
350 		return (-1);
351 	}
352 
353 	/* Expect Vendor Specific Event 0x06 */
354 	event = (struct iwmbt_hci_event *)buf;
355 	if (event->header.event != 0xFF || event->data[0] != 0x06) {
356 		iwmbt_err("firmware download completion event missed");
357 		return (-1);
358 	}
359 
360 	return (0);
361 }
362 
363 int
364 iwmbt_enter_manufacturer(struct libusb_device_handle *hdl)
365 {
366 	int ret, transferred;
367 	static struct iwmbt_hci_cmd cmd = {
368 		.opcode = htole16(0xfc11),
369 		.length = 2,
370 		.data = { 0x01, 0x00 },
371 	};
372 	uint8_t buf[IWMBT_HCI_MAX_EVENT_SIZE];
373 
374 	ret = iwmbt_hci_command(hdl,
375 	    &cmd,
376 	    buf,
377 	    sizeof(buf),
378 	    &transferred,
379 	    IWMBT_HCI_CMD_TIMEOUT);
380 
381 	if (ret < 0) {
382 		 iwmbt_debug("Can't enter manufacturer mode: code=%d, size=%d",
383 		     ret,
384 		     transferred);
385 		 return (-1);
386 	}
387 
388 	return (0);
389 }
390 
391 int
392 iwmbt_exit_manufacturer(struct libusb_device_handle *hdl, int mode)
393 {
394 	int ret, transferred;
395 	static struct iwmbt_hci_cmd cmd = {
396 		.opcode = htole16(0xfc11),
397 		.length = 2,
398 		.data = { 0x00, 0x00 },
399 	};
400 	uint8_t buf[IWMBT_HCI_MAX_EVENT_SIZE];
401 
402 	/*
403 	 * The mode sets the type of reset we want to perform:
404 	 * 0x00: simply exit manufacturer mode without a reset.
405 	 * 0x01: exit manufacturer mode with a reset and patches disabled
406 	 * 0x02: exit manufacturer mode with a reset and patches enabled
407 	 */
408 	if (mode > 2) {
409 		iwmbt_debug("iwmbt_exit_manufacturer(): unknown mode (%d)",
410 				mode);
411 	}
412 	cmd.data[1] = mode;
413 
414 	ret = iwmbt_hci_command(hdl,
415 	    &cmd,
416 	    buf,
417 	    sizeof(buf),
418 	    &transferred,
419 	    IWMBT_HCI_CMD_TIMEOUT);
420 
421 	if (ret < 0) {
422 		 iwmbt_debug("Can't exit manufacturer mode: code=%d, size=%d",
423 		     ret,
424 		     transferred);
425 		 return (-1);
426 	}
427 
428 	return (0);
429 }
430 
431 int
432 iwmbt_get_version(struct libusb_device_handle *hdl,
433     struct iwmbt_version *version)
434 {
435 	int ret, transferred;
436 	struct iwmbt_hci_event_cmd_compl*event;
437 	struct iwmbt_hci_cmd cmd = {
438 		.opcode = htole16(0xfc05),
439 		.length = 0,
440 	};
441 	uint8_t buf[IWMBT_HCI_EVT_COMPL_SIZE(struct iwmbt_version)];
442 
443 	memset(buf, 0, sizeof(buf));
444 
445 	ret = iwmbt_hci_command(hdl,
446 	    &cmd,
447 	    buf,
448 	    sizeof(buf),
449 	    &transferred,
450 	    IWMBT_HCI_CMD_TIMEOUT);
451 
452 	if (ret < 0 || transferred != sizeof(buf)) {
453 		 iwmbt_debug("Can't get version: : code=%d, size=%d",
454 		     ret,
455 		     transferred);
456 		 return (-1);
457 	}
458 
459 	event = (struct iwmbt_hci_event_cmd_compl *)buf;
460 	memcpy(version, event->data, sizeof(struct iwmbt_version));
461 
462 	return (0);
463 }
464 
465 int
466 iwmbt_get_boot_params(struct libusb_device_handle *hdl,
467     struct iwmbt_boot_params *params)
468 {
469 	int ret, transferred = 0;
470 	struct iwmbt_hci_event_cmd_compl *event;
471 	struct iwmbt_hci_cmd cmd = {
472 		.opcode = htole16(0xfc0d),
473 		.length = 0,
474 	};
475 	uint8_t buf[IWMBT_HCI_EVT_COMPL_SIZE(struct iwmbt_boot_params)];
476 
477 	memset(buf, 0, sizeof(buf));
478 
479 	ret = iwmbt_hci_command(hdl,
480 	    &cmd,
481 	    buf,
482 	    sizeof(buf),
483 	    &transferred,
484 	    IWMBT_HCI_CMD_TIMEOUT);
485 
486 	if (ret < 0 || transferred != sizeof(buf)) {
487 		 iwmbt_debug("Can't get boot params: code=%d, size=%d",
488 		     ret,
489 		     transferred);
490 		 return (-1);
491 	}
492 
493 	event = (struct iwmbt_hci_event_cmd_compl *)buf;
494 	memcpy(params, event->data, sizeof(struct iwmbt_boot_params));
495 
496 	return (0);
497 }
498 
499 int
500 iwmbt_intel_reset(struct libusb_device_handle *hdl, uint32_t boot_param)
501 {
502 	int ret, transferred = 0;
503 	struct iwmbt_hci_event *event;
504 	static struct iwmbt_hci_cmd cmd = {
505 		.opcode = htole16(0xfc01),
506 		.length = 8,
507 		.data = { 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00 },
508 	};
509 	uint8_t buf[IWMBT_HCI_MAX_EVENT_SIZE];
510 
511 	le32enc(cmd.data + 4, boot_param);
512 	memset(buf, 0, sizeof(buf));
513 
514 	ret = iwmbt_hci_command(hdl,
515 	    &cmd,
516 	    buf,
517 	    sizeof(buf),
518 	    &transferred,
519 	    IWMBT_HCI_CMD_TIMEOUT);
520 
521 	if (ret < 0 || transferred < (int)sizeof(struct iwmbt_hci_event) + 1) {
522 		 iwmbt_debug("Intel Reset command failed: code=%d, size=%d",
523 		    ret,
524 		    transferred);
525 		 return (ret);
526 	}
527 
528 	/* expect Vendor Specific Event 0x02 */
529 	event = (struct iwmbt_hci_event *)buf;
530 	if (event->header.event != 0xFF || event->data[0] != 0x02) {
531 		iwmbt_err("Intel Reset completion event missed");
532 		return (-1);
533 	}
534 
535 	return (0);
536 }
537 
538 int
539 iwmbt_load_ddc(struct libusb_device_handle *hdl,
540     const struct iwmbt_firmware *ddc)
541 {
542 	int size, sent = 0;
543 	int ret, transferred;
544 	uint8_t buf[IWMBT_HCI_MAX_CMD_SIZE];
545 	struct iwmbt_hci_cmd *cmd = (struct iwmbt_hci_cmd *)buf;
546 
547 	size = ddc->len;
548 
549 	iwmbt_debug("file=%s, size=%d", ddc->fwname, size);
550 
551 	while (size > 0) {
552 
553 		memset(buf, 0, sizeof(buf));
554 		cmd->opcode = htole16(0xfc8b);
555 		cmd->length = ddc->buf[sent] + 1;
556 		memcpy(cmd->data, ddc->buf + sent, XMIN(ddc->buf[sent], size));
557 
558 		iwmbt_debug("transferring %d bytes, offset %d",
559 		    cmd->length,
560 		    sent);
561 
562 		size -= cmd->length;
563 		sent += cmd->length;
564 
565 		ret = iwmbt_hci_command(hdl,
566 		    cmd,
567 		    buf,
568 		    sizeof(buf),
569 		    &transferred,
570 		    IWMBT_HCI_CMD_TIMEOUT);
571 
572 		if (ret < 0) {
573 			 iwmbt_debug("Intel Write DDC failed: code=%d", ret);
574 			 return (-1);
575 		}
576 	}
577 
578 	return (0);
579 }
580 
581 int
582 iwmbt_set_event_mask(struct libusb_device_handle *hdl)
583 {
584 	int ret, transferred = 0;
585 	static struct iwmbt_hci_cmd cmd = {
586 		.opcode = htole16(0xfc52),
587 		.length = 8,
588 		.data = { 0x87, 0x0c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 },
589 	};
590 	uint8_t buf[IWMBT_HCI_MAX_EVENT_SIZE];
591 
592 	ret = iwmbt_hci_command(hdl,
593 	    &cmd,
594 	    buf,
595 	    sizeof(buf),
596 	    &transferred,
597 	    IWMBT_HCI_CMD_TIMEOUT);
598 
599 	if (ret < 0)
600 		 iwmbt_debug("Intel Set Event Mask failed: code=%d", ret);
601 
602 	return (ret);
603 }
604