Linux Kernel  3.7.1
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
mac.c
Go to the documentation of this file.
1 /*
2  * Intel Wireless UWB Link 1480
3  * MAC Firmware upload implementation
4  *
5  * Copyright (C) 2005-2006 Intel Corporation
6  * Inaky Perez-Gonzalez <[email protected]>
7  *
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.
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 Street, Fifth Floor, Boston, MA
20  * 02110-1301, USA.
21  *
22  *
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.
28  */
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"
34 
35 /*
36  * Descriptor for a continuous segment of MAC fw data
37  */
38 struct fw_hdr {
39  unsigned long address;
40  size_t length;
41  const u32 *bin;
42  struct fw_hdr *next;
43 };
44 
45 
46 /* Free a chain of firmware headers */
47 static
48 void fw_hdrs_free(struct fw_hdr *hdr)
49 {
50  struct fw_hdr *next;
51 
52  while (hdr) {
53  next = hdr->next;
54  kfree(hdr);
55  hdr = next;
56  }
57 }
58 
59 
60 /* Fill a firmware header descriptor from a memory buffer */
61 static
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)
64 {
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",
70  hdr_cnt, hdr_offset,
71  (const char *) data_itr + 2 - _data,
72  (const char *) data_top - _data);
73  return -EINVAL;
74  }
75  hdr->next = NULL;
76  hdr->address = le32_to_cpu(*data_itr++);
77  hdr->length = le32_to_cpu(*data_itr++);
78  hdr->bin = 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);
83  return -EINVAL;
84  }
85  return 0;
86 }
87 
88 
107 static
108 int fw_hdrs_load(struct i1480 *i1480, struct fw_hdr **phdr,
109  const char *_data, size_t data_size)
110 {
111  int result;
112  unsigned hdr_cnt = 0;
113  u32 *data = (u32 *) _data, *data_itr, *data_top;
114  struct fw_hdr *hdr, **prev_hdr = phdr;
115 
116  result = -EINVAL;
117  /* Check size is ok and pointer is aligned */
118  if (data_size % sizeof(u32) != 0)
119  goto error;
120  if ((unsigned long) _data % sizeof(u16) != 0)
121  goto error;
122  *phdr = NULL;
123  data_itr = data;
124  data_top = (u32 *) (_data + data_size);
125  while (data_itr < data_top) {
126  result = -ENOMEM;
127  hdr = kmalloc(sizeof(*hdr), GFP_KERNEL);
128  if (hdr == NULL) {
129  dev_err(i1480->dev, "Cannot allocate fw header "
130  "for chunk #%u\n", hdr_cnt);
131  goto error_alloc;
132  }
133  result = fw_hdr_load(i1480, hdr, hdr_cnt,
134  _data, data_itr, data_top);
135  if (result < 0)
136  goto error_load;
137  data_itr += 2 + hdr->length;
138  *prev_hdr = hdr;
139  prev_hdr = &hdr->next;
140  hdr_cnt++;
141  };
142  *prev_hdr = NULL;
143  return 0;
144 
145 error_load:
146  kfree(hdr);
147 error_alloc:
148  fw_hdrs_free(*phdr);
149 error:
150  return result;
151 }
152 
153 
166 static
167 ssize_t i1480_fw_cmp(struct i1480 *i1480, struct fw_hdr *hdr)
168 {
169  ssize_t result = 0;
170  u32 src_itr = 0, cnt;
171  size_t size = hdr->length*sizeof(hdr->bin[0]);
172  size_t chunk_size;
173  u8 *bin = (u8 *) hdr->bin;
174 
175  while (size > 0) {
176  chunk_size = size < i1480->buf_size ? size : i1480->buf_size;
177  result = i1480->read(i1480, hdr->address + src_itr, chunk_size);
178  if (result < 0) {
179  dev_err(i1480->dev, "error reading for verification: "
180  "%zd\n", result);
181  goto error;
182  }
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;
192  goto cmp_failed;
193  }
194  }
195  src_itr += result;
196  size -= result;
197  }
198  result = 0;
199 error:
200 cmp_failed:
201  return result;
202 }
203 
204 
212 static
213 int mac_fw_hdrs_push(struct i1480 *i1480, struct fw_hdr *hdr,
214  const char *fw_name, const char *fw_tag)
215 {
216  struct device *dev = i1480->dev;
217  ssize_t result = 0;
218  struct fw_hdr *hdr_itr;
219  int verif_retry_count;
220 
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;
224 retry:
225  dev_dbg(dev, "fw chunk (%zu @ 0x%08lx)\n",
226  hdr_itr->length * sizeof(hdr_itr->bin[0]),
227  hdr_itr->address);
228  result = i1480->write(i1480, hdr_itr->address, hdr_itr->bin,
229  hdr_itr->length*sizeof(hdr_itr->bin[0]));
230  if (result < 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);
235  break;
236  }
237  result = i1480_fw_cmp(i1480, hdr_itr);
238  if (result < 0) {
239  dev_err(dev, "%s fw '%s': verification read "
240  "failed (%zuB @ 0x%lx): %zd\n",
241  fw_tag, fw_name,
242  hdr_itr->length * sizeof(hdr_itr->bin[0]),
243  hdr_itr->address, result);
244  break;
245  }
246  if (result > 0) { /* Offset where it failed + 1 */
247  result--;
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);
256  result = -EINVAL;
257  break;
258  }
259  }
260  return result;
261 }
262 
263 
265 static
266 int mac_fw_upload_enable(struct i1480 *i1480)
267 {
268  int result;
269  u32 reg = 0x800000c0;
270  u32 *buffer = (u32 *)i1480->cmd_buf;
271 
272  if (i1480->hw_rev > 1)
273  reg = 0x8000d0d4;
274  result = i1480->read(i1480, reg, sizeof(u32));
275  if (result < 0)
276  goto error_cmd;
277  *buffer &= ~i1480_FW_UPLOAD_MODE_MASK;
278  result = i1480->write(i1480, reg, buffer, sizeof(u32));
279  if (result < 0)
280  goto error_cmd;
281  return 0;
282 error_cmd:
283  dev_err(i1480->dev, "can't enable fw upload mode: %d\n", result);
284  return result;
285 }
286 
287 
289 static
290 int mac_fw_upload_disable(struct i1480 *i1480)
291 {
292  int result;
293  u32 reg = 0x800000c0;
294  u32 *buffer = (u32 *)i1480->cmd_buf;
295 
296  if (i1480->hw_rev > 1)
297  reg = 0x8000d0d4;
298  result = i1480->read(i1480, reg, sizeof(u32));
299  if (result < 0)
300  goto error_cmd;
301  *buffer |= i1480_FW_UPLOAD_MODE_MASK;
302  result = i1480->write(i1480, reg, buffer, sizeof(u32));
303  if (result < 0)
304  goto error_cmd;
305  return 0;
306 error_cmd:
307  dev_err(i1480->dev, "can't disable fw upload mode: %d\n", result);
308  return result;
309 }
310 
311 
312 
324 static
325 int __mac_fw_upload(struct i1480 *i1480, const char *fw_name,
326  const char *fw_tag)
327 {
328  int result;
329  const struct firmware *fw;
330  struct fw_hdr *fw_hdrs;
331 
332  result = request_firmware(&fw, fw_name, i1480->dev);
333  if (result < 0) /* Up to caller to complain on -ENOENT */
334  goto out;
335  result = fw_hdrs_load(i1480, &fw_hdrs, fw->data, fw->size);
336  if (result < 0) {
337  dev_err(i1480->dev, "%s fw '%s': failed to parse firmware "
338  "file: %d\n", fw_tag, fw_name, result);
339  goto out_release;
340  }
341  result = mac_fw_upload_enable(i1480);
342  if (result < 0)
343  goto out_hdrs_release;
344  result = mac_fw_hdrs_push(i1480, fw_hdrs, fw_name, fw_tag);
345  mac_fw_upload_disable(i1480);
346 out_hdrs_release:
347  if (result >= 0)
348  dev_info(i1480->dev, "%s fw '%s': uploaded\n", fw_tag, fw_name);
349  else
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);
353 out_release:
354  release_firmware(fw);
355 out:
356  return result;
357 }
358 
359 
364 int i1480_pre_fw_upload(struct i1480 *i1480)
365 {
366  int result;
367  result = __mac_fw_upload(i1480, i1480->pre_fw_name, "PRE");
368  if (result == 0)
369  msleep(400);
370  return result;
371 }
372 
373 
387 static
388 int i1480_cmd_reset(struct i1480 *i1480)
389 {
390  int result;
391  struct uwb_rccb *cmd = (void *) i1480->cmd_buf;
392  struct i1480_evt_reset {
393  struct uwb_rceb rceb;
394  u8 bResultCode;
395  } __attribute__((packed)) *reply = (void *) i1480->evt_buf;
396 
397  result = -ENOMEM;
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));
403  if (result < 0)
404  goto out;
405  if (reply->bResultCode != UWB_RC_RES_SUCCESS) {
406  dev_err(i1480->dev, "RESET: command execution failed: %u\n",
407  reply->bResultCode);
408  result = -EIO;
409  }
410 out:
411  return result;
412 
413 }
414 
415 
416 /* Wait for the MAC FW to start running */
417 static
418 int i1480_fw_is_running_q(struct i1480 *i1480)
419 {
420  int cnt = 0;
421  int result;
422  u32 *val = (u32 *) i1480->cmd_buf;
423 
424  for (cnt = 0; cnt < 10; cnt++) {
425  msleep(100);
426  result = i1480->read(i1480, 0x80080000, 4);
427  if (result < 0) {
428  dev_err(i1480->dev, "Can't read 0x8008000: %d\n", result);
429  goto out;
430  }
431  if (*val == 0x55555555UL) /* fw running? cool */
432  goto out;
433  }
434  dev_err(i1480->dev, "Timed out waiting for fw to start\n");
435  result = -ETIMEDOUT;
436 out:
437  return result;
438 
439 }
440 
441 
451 int i1480_mac_fw_upload(struct i1480 *i1480)
452 {
453  int result = 0, deprecated_name = 0;
454  struct i1480_rceb *rcebe = (void *) i1480->evt_buf;
455 
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,
459  "MAC");
460  deprecated_name = 1;
461  }
462  if (result < 0)
463  return result;
464  if (deprecated_name == 1)
465  dev_warn(i1480->dev,
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);
470  if (result < 0)
471  goto error_fw_not_running;
472  result = i1480->rc_setup ? i1480->rc_setup(i1480) : 0;
473  if (result < 0) {
474  dev_err(i1480->dev, "Cannot setup after MAC fw upload: %d\n",
475  result);
476  goto error_setup;
477  }
478  result = i1480->wait_init_done(i1480); /* wait init'on */
479  if (result < 0) {
480  dev_err(i1480->dev, "MAC fw '%s': Initialization timed out "
481  "(%d)\n", i1480->mac_fw_name, result);
482  goto error_init_timeout;
483  }
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));
489  goto error_size;
490  }
491  result = -EIO;
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,
499  goto error_init_timeout;
500  }
501  result = i1480_cmd_reset(i1480);
502  if (result < 0)
503  dev_err(i1480->dev, "MAC fw '%s': MBOA reset failed (%d)\n",
504  i1480->mac_fw_name, result);
505 error_fw_not_running:
506 error_init_timeout:
507 error_size:
508 error_setup:
509  return result;
510 }