2 * Intel Wireless UWB Link 1480
3 * MAC Firmware upload implementation
5 * Copyright (C) 2005-2006 Intel Corporation
6 * Inaky Perez-Gonzalez <inaky.perez-gonzalez@intel.com>
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License version
10 * 2 as published by the Free Software Foundation.
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.
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 Street, Fifth Floor, Boston, MA
23 * Implementation of the code for parsing the firmware file (extract
24 * the headers and binary code chunks) in the fw_*() functions. The
25 * code to upload pre and mac firmwares is the same, so it uses a
26 * common entry point in __mac_fw_upload(), which uses the i1480
27 * function pointers to push the firmware to the device.
29 #include <linux/delay.h>
30 #include <linux/firmware.h>
31 #include <linux/slab.h>
32 #include <linux/uwb.h>
33 #include "i1480-dfu.h"
36 * Descriptor for a continuous segment of MAC fw data
39 unsigned long address
;
46 /* Free a chain of firmware headers */
48 void fw_hdrs_free(struct fw_hdr
*hdr
)
60 /* Fill a firmware header descriptor from a memory buffer */
62 int fw_hdr_load(struct i1480
*i1480
, struct fw_hdr
*hdr
, unsigned hdr_cnt
,
63 const char *_data
, const u32
*data_itr
, const u32
*data_top
)
65 size_t hdr_offset
= (const char *) data_itr
- _data
;
66 size_t remaining_size
= (void *) data_top
- (void *) data_itr
;
67 if (data_itr
+ 2 > data_top
) {
68 dev_err(i1480
->dev
, "fw hdr #%u/%zu: EOF reached in header at "
69 "offset %zu, limit %zu\n",
71 (const char *) data_itr
+ 2 - _data
,
72 (const char *) data_top
- _data
);
76 hdr
->address
= le32_to_cpu(*data_itr
++);
77 hdr
->length
= le32_to_cpu(*data_itr
++);
79 if (hdr
->length
> remaining_size
) {
80 dev_err(i1480
->dev
, "fw hdr #%u/%zu: EOF reached in data; "
81 "chunk too long (%zu bytes), only %zu left\n",
82 hdr_cnt
, hdr_offset
, hdr
->length
, remaining_size
);
90 * Get a buffer where the firmware is supposed to be and create a
91 * chain of headers linking them together.
93 * @phdr: where to place the pointer to the first header (headers link
94 * to the next via the @hdr->next ptr); need to free the whole
97 * @_data: Pointer to the data buffer.
99 * @_data_size: Size of the data buffer (bytes); data size has to be a
100 * multiple of 4. Function will fail if not.
102 * Goes over the whole binary blob; reads the first chunk and creates
103 * a fw hdr from it (which points to where the data is in @_data and
104 * the length of the chunk); then goes on to the next chunk until
105 * done. Each header is linked to the next.
108 int fw_hdrs_load(struct i1480
*i1480
, struct fw_hdr
**phdr
,
109 const char *_data
, size_t data_size
)
112 unsigned hdr_cnt
= 0;
113 u32
*data
= (u32
*) _data
, *data_itr
, *data_top
;
114 struct fw_hdr
*hdr
, **prev_hdr
= phdr
;
117 /* Check size is ok and pointer is aligned */
118 if (data_size
% sizeof(u32
) != 0)
120 if ((unsigned long) _data
% sizeof(u16
) != 0)
124 data_top
= (u32
*) (_data
+ data_size
);
125 while (data_itr
< data_top
) {
127 hdr
= kmalloc(sizeof(*hdr
), GFP_KERNEL
);
129 dev_err(i1480
->dev
, "Cannot allocate fw header "
130 "for chunk #%u\n", hdr_cnt
);
133 result
= fw_hdr_load(i1480
, hdr
, hdr_cnt
,
134 _data
, data_itr
, data_top
);
137 data_itr
+= 2 + hdr
->length
;
139 prev_hdr
= &hdr
->next
;
155 * Compares a chunk of fw with one in the devices's memory
157 * @i1480: Device instance
158 * @hdr: Pointer to the firmware chunk
159 * @returns: 0 if equal, < 0 errno on error. If > 0, it is the offset
160 * where the difference was found (plus one).
162 * Kind of dirty and simplistic, but does the trick in both the PCI
163 * and USB version. We do a quick[er] memcmp(), and if it fails, we do
164 * a byte-by-byte to find the offset.
167 ssize_t
i1480_fw_cmp(struct i1480
*i1480
, struct fw_hdr
*hdr
)
170 u32 src_itr
= 0, cnt
;
171 size_t size
= hdr
->length
*sizeof(hdr
->bin
[0]);
173 u8
*bin
= (u8
*) hdr
->bin
;
176 chunk_size
= size
< i1480
->buf_size
? size
: i1480
->buf_size
;
177 result
= i1480
->read(i1480
, hdr
->address
+ src_itr
, chunk_size
);
179 dev_err(i1480
->dev
, "error reading for verification: "
183 if (memcmp(i1480
->cmd_buf
, bin
+ src_itr
, result
)) {
184 u8
*buf
= i1480
->cmd_buf
;
185 for (cnt
= 0; cnt
< result
; cnt
++)
186 if (bin
[src_itr
+ cnt
] != buf
[cnt
]) {
187 dev_err(i1480
->dev
, "byte failed at "
188 "src_itr %u cnt %u [0x%02x "
189 "vs 0x%02x]\n", src_itr
, cnt
,
190 bin
[src_itr
+ cnt
], buf
[cnt
]);
191 result
= src_itr
+ cnt
+ 1;
206 * Writes firmware headers to the device.
209 * @hdr: Processed firmware
210 * @returns: 0 if ok, < 0 errno on error.
213 int mac_fw_hdrs_push(struct i1480
*i1480
, struct fw_hdr
*hdr
,
214 const char *fw_name
, const char *fw_tag
)
216 struct device
*dev
= i1480
->dev
;
218 struct fw_hdr
*hdr_itr
;
219 int verif_retry_count
;
221 /* Now, header by header, push them to the hw */
222 for (hdr_itr
= hdr
; hdr_itr
!= NULL
; hdr_itr
= hdr_itr
->next
) {
223 verif_retry_count
= 0;
225 dev_dbg(dev
, "fw chunk (%zu @ 0x%08lx)\n",
226 hdr_itr
->length
* sizeof(hdr_itr
->bin
[0]),
228 result
= i1480
->write(i1480
, hdr_itr
->address
, hdr_itr
->bin
,
229 hdr_itr
->length
*sizeof(hdr_itr
->bin
[0]));
231 dev_err(dev
, "%s fw '%s': write failed (%zuB @ 0x%lx):"
232 " %zd\n", fw_tag
, fw_name
,
233 hdr_itr
->length
* sizeof(hdr_itr
->bin
[0]),
234 hdr_itr
->address
, result
);
237 result
= i1480_fw_cmp(i1480
, hdr_itr
);
239 dev_err(dev
, "%s fw '%s': verification read "
240 "failed (%zuB @ 0x%lx): %zd\n",
242 hdr_itr
->length
* sizeof(hdr_itr
->bin
[0]),
243 hdr_itr
->address
, result
);
246 if (result
> 0) { /* Offset where it failed + 1 */
248 dev_err(dev
, "%s fw '%s': WARNING: verification "
249 "failed at 0x%lx: retrying\n",
250 fw_tag
, fw_name
, hdr_itr
->address
+ result
);
251 if (++verif_retry_count
< 3)
252 goto retry
; /* write this block again! */
253 dev_err(dev
, "%s fw '%s': verification failed at 0x%lx: "
254 "tried %d times\n", fw_tag
, fw_name
,
255 hdr_itr
->address
+ result
, verif_retry_count
);
264 /** Puts the device in firmware upload mode.*/
266 int mac_fw_upload_enable(struct i1480
*i1480
)
269 u32 reg
= 0x800000c0;
270 u32
*buffer
= (u32
*)i1480
->cmd_buf
;
272 if (i1480
->hw_rev
> 1)
274 result
= i1480
->read(i1480
, reg
, sizeof(u32
));
277 *buffer
&= ~i1480_FW_UPLOAD_MODE_MASK
;
278 result
= i1480
->write(i1480
, reg
, buffer
, sizeof(u32
));
283 dev_err(i1480
->dev
, "can't enable fw upload mode: %d\n", result
);
288 /** Gets the device out of firmware upload mode. */
290 int mac_fw_upload_disable(struct i1480
*i1480
)
293 u32 reg
= 0x800000c0;
294 u32
*buffer
= (u32
*)i1480
->cmd_buf
;
296 if (i1480
->hw_rev
> 1)
298 result
= i1480
->read(i1480
, reg
, sizeof(u32
));
301 *buffer
|= i1480_FW_UPLOAD_MODE_MASK
;
302 result
= i1480
->write(i1480
, reg
, buffer
, sizeof(u32
));
307 dev_err(i1480
->dev
, "can't disable fw upload mode: %d\n", result
);
314 * Generic function for uploading a MAC firmware.
316 * @i1480: Device instance
317 * @fw_name: Name of firmware file to upload.
318 * @fw_tag: Name of the firmware type (for messages)
320 * @do_wait: Wait for device to emit initialization done message (0
321 * for PRE fws, 1 for MAC fws).
322 * @returns: 0 if ok, < 0 errno on error.
325 int __mac_fw_upload(struct i1480
*i1480
, const char *fw_name
,
329 const struct firmware
*fw
;
330 struct fw_hdr
*fw_hdrs
;
332 result
= request_firmware(&fw
, fw_name
, i1480
->dev
);
333 if (result
< 0) /* Up to caller to complain on -ENOENT */
335 result
= fw_hdrs_load(i1480
, &fw_hdrs
, fw
->data
, fw
->size
);
337 dev_err(i1480
->dev
, "%s fw '%s': failed to parse firmware "
338 "file: %d\n", fw_tag
, fw_name
, result
);
341 result
= mac_fw_upload_enable(i1480
);
343 goto out_hdrs_release
;
344 result
= mac_fw_hdrs_push(i1480
, fw_hdrs
, fw_name
, fw_tag
);
345 mac_fw_upload_disable(i1480
);
348 dev_info(i1480
->dev
, "%s fw '%s': uploaded\n", fw_tag
, fw_name
);
350 dev_err(i1480
->dev
, "%s fw '%s': failed to upload (%d), "
351 "power cycle device\n", fw_tag
, fw_name
, result
);
352 fw_hdrs_free(fw_hdrs
);
354 release_firmware(fw
);
361 * Upload a pre-PHY firmware
364 int i1480_pre_fw_upload(struct i1480
*i1480
)
367 result
= __mac_fw_upload(i1480
, i1480
->pre_fw_name
, "PRE");
375 * Reset a the MAC and PHY
377 * @i1480: Device's instance
378 * @returns: 0 if ok, < 0 errno code on error
380 * We put the command on kmalloc'ed memory as some arches cannot do
381 * USB from the stack. The reply event is copied from an stage buffer,
382 * so it can be in the stack. See WUSB1.0[8.6.2.4] for more details.
384 * We issue the reset to make sure the UWB controller reinits the PHY;
385 * this way we can now if the PHY init went ok.
388 int i1480_cmd_reset(struct i1480
*i1480
)
391 struct uwb_rccb
*cmd
= (void *) i1480
->cmd_buf
;
392 struct i1480_evt_reset
{
393 struct uwb_rceb rceb
;
395 } __attribute__((packed
)) *reply
= (void *) i1480
->evt_buf
;
398 cmd
->bCommandType
= UWB_RC_CET_GENERAL
;
399 cmd
->wCommand
= cpu_to_le16(UWB_RC_CMD_RESET
);
400 reply
->rceb
.bEventType
= UWB_RC_CET_GENERAL
;
401 reply
->rceb
.wEvent
= UWB_RC_CMD_RESET
;
402 result
= i1480_cmd(i1480
, "RESET", sizeof(*cmd
), sizeof(*reply
));
405 if (reply
->bResultCode
!= UWB_RC_RES_SUCCESS
) {
406 dev_err(i1480
->dev
, "RESET: command execution failed: %u\n",
416 /* Wait for the MAC FW to start running */
418 int i1480_fw_is_running_q(struct i1480
*i1480
)
422 u32
*val
= (u32
*) i1480
->cmd_buf
;
424 for (cnt
= 0; cnt
< 10; cnt
++) {
426 result
= i1480
->read(i1480
, 0x80080000, 4);
428 dev_err(i1480
->dev
, "Can't read 0x8008000: %d\n", result
);
431 if (*val
== 0x55555555UL
) /* fw running? cool */
434 dev_err(i1480
->dev
, "Timed out waiting for fw to start\n");
443 * Upload MAC firmware, wait for it to start
445 * @i1480: Device instance
446 * @fw_name: Name of the file that contains the firmware
448 * This has to be called after the pre fw has been uploaded (if
451 int i1480_mac_fw_upload(struct i1480
*i1480
)
453 int result
= 0, deprecated_name
= 0;
454 struct i1480_rceb
*rcebe
= (void *) i1480
->evt_buf
;
456 result
= __mac_fw_upload(i1480
, i1480
->mac_fw_name
, "MAC");
457 if (result
== -ENOENT
) {
458 result
= __mac_fw_upload(i1480
, i1480
->mac_fw_name_deprecate
,
464 if (deprecated_name
== 1)
466 "WARNING: firmware file name %s is deprecated, "
467 "please rename to %s\n",
468 i1480
->mac_fw_name_deprecate
, i1480
->mac_fw_name
);
469 result
= i1480_fw_is_running_q(i1480
);
471 goto error_fw_not_running
;
472 result
= i1480
->rc_setup
? i1480
->rc_setup(i1480
) : 0;
474 dev_err(i1480
->dev
, "Cannot setup after MAC fw upload: %d\n",
478 result
= i1480
->wait_init_done(i1480
); /* wait init'on */
480 dev_err(i1480
->dev
, "MAC fw '%s': Initialization timed out "
481 "(%d)\n", i1480
->mac_fw_name
, result
);
482 goto error_init_timeout
;
484 /* verify we got the right initialization done event */
485 if (i1480
->evt_result
!= sizeof(*rcebe
)) {
486 dev_err(i1480
->dev
, "MAC fw '%s': initialization event returns "
487 "wrong size (%zu bytes vs %zu needed)\n",
488 i1480
->mac_fw_name
, i1480
->evt_result
, sizeof(*rcebe
));
492 if (i1480_rceb_check(i1480
, &rcebe
->rceb
, NULL
, 0, i1480_CET_VS1
,
493 i1480_EVT_RM_INIT_DONE
) < 0) {
494 dev_err(i1480
->dev
, "wrong initialization event 0x%02x/%04x/%02x "
495 "received; expected 0x%02x/%04x/00\n",
496 rcebe
->rceb
.bEventType
, le16_to_cpu(rcebe
->rceb
.wEvent
),
497 rcebe
->rceb
.bEventContext
, i1480_CET_VS1
,
498 i1480_EVT_RM_INIT_DONE
);
499 goto error_init_timeout
;
501 result
= i1480_cmd_reset(i1480
);
503 dev_err(i1480
->dev
, "MAC fw '%s': MBOA reset failed (%d)\n",
504 i1480
->mac_fw_name
, result
);
505 error_fw_not_running
: