2 * Copyright (c) 2004-2011 Atheros Communications Inc.
3 * Copyright (c) 2011-2012 Qualcomm Atheros, Inc.
5 * Permission to use, copy, modify, and/or distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
23 int ath6kl_bmi_done(struct ath6kl
*ar
)
28 if (ar
->bmi
.done_sent
) {
29 ath6kl_dbg(ATH6KL_DBG_BMI
, "bmi done skipped\n");
33 ar
->bmi
.done_sent
= true;
35 ret
= ath6kl_hif_bmi_write(ar
, (u8
*)&cid
, sizeof(cid
));
37 ath6kl_err("Unable to send bmi done: %d\n", ret
);
44 int ath6kl_bmi_get_target_info(struct ath6kl
*ar
,
45 struct ath6kl_bmi_target_info
*targ_info
)
48 u32 cid
= BMI_GET_TARGET_INFO
;
50 if (ar
->bmi
.done_sent
) {
51 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
55 ret
= ath6kl_hif_bmi_write(ar
, (u8
*)&cid
, sizeof(cid
));
57 ath6kl_err("Unable to send get target info: %d\n", ret
);
61 if (ar
->hif_type
== ATH6KL_HIF_TYPE_USB
) {
62 ret
= ath6kl_hif_bmi_read(ar
, (u8
*)targ_info
,
65 ret
= ath6kl_hif_bmi_read(ar
, (u8
*)&targ_info
->version
,
66 sizeof(targ_info
->version
));
70 ath6kl_err("Unable to recv target info: %d\n", ret
);
74 if (le32_to_cpu(targ_info
->version
) == TARGET_VERSION_SENTINAL
) {
75 /* Determine how many bytes are in the Target's targ_info */
76 ret
= ath6kl_hif_bmi_read(ar
,
77 (u8
*)&targ_info
->byte_count
,
78 sizeof(targ_info
->byte_count
));
80 ath6kl_err("unable to read target info byte count: %d\n",
86 * The target's targ_info doesn't match the host's targ_info.
87 * We need to do some backwards compatibility to make this work.
89 if (le32_to_cpu(targ_info
->byte_count
) != sizeof(*targ_info
)) {
94 /* Read the remainder of the targ_info */
95 ret
= ath6kl_hif_bmi_read(ar
,
97 sizeof(targ_info
->byte_count
),
99 sizeof(targ_info
->byte_count
));
102 ath6kl_err("Unable to read target info (%d bytes): %d\n",
103 targ_info
->byte_count
, ret
);
108 ath6kl_dbg(ATH6KL_DBG_BMI
, "target info (ver: 0x%x type: 0x%x)\n",
109 targ_info
->version
, targ_info
->type
);
114 int ath6kl_bmi_read(struct ath6kl
*ar
, u32 addr
, u8
*buf
, u32 len
)
116 u32 cid
= BMI_READ_MEMORY
;
119 u32 len_remain
, rx_len
;
122 if (ar
->bmi
.done_sent
) {
123 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
127 size
= ar
->bmi
.max_data_size
+ sizeof(cid
) + sizeof(addr
) + sizeof(len
);
128 if (size
> ar
->bmi
.max_cmd_size
) {
132 memset(ar
->bmi
.cmd_buf
, 0, size
);
134 ath6kl_dbg(ATH6KL_DBG_BMI
,
135 "bmi read memory: device: addr: 0x%x, len: %d\n",
141 rx_len
= (len_remain
< ar
->bmi
.max_data_size
) ?
142 len_remain
: ar
->bmi
.max_data_size
;
144 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &cid
, sizeof(cid
));
145 offset
+= sizeof(cid
);
146 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &addr
, sizeof(addr
));
147 offset
+= sizeof(addr
);
148 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &rx_len
, sizeof(rx_len
));
149 offset
+= sizeof(len
);
151 ret
= ath6kl_hif_bmi_write(ar
, ar
->bmi
.cmd_buf
, offset
);
153 ath6kl_err("Unable to write to the device: %d\n",
157 ret
= ath6kl_hif_bmi_read(ar
, ar
->bmi
.cmd_buf
, rx_len
);
159 ath6kl_err("Unable to read from the device: %d\n",
163 memcpy(&buf
[len
- len_remain
], ar
->bmi
.cmd_buf
, rx_len
);
164 len_remain
-= rx_len
; addr
+= rx_len
;
170 int ath6kl_bmi_write(struct ath6kl
*ar
, u32 addr
, u8
*buf
, u32 len
)
172 u32 cid
= BMI_WRITE_MEMORY
;
175 u32 len_remain
, tx_len
;
176 const u32 header
= sizeof(cid
) + sizeof(addr
) + sizeof(len
);
180 if (ar
->bmi
.done_sent
) {
181 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
185 if ((ar
->bmi
.max_data_size
+ header
) > ar
->bmi
.max_cmd_size
) {
190 if (WARN_ON(ar
->bmi
.max_data_size
> sizeof(aligned_buf
)))
193 memset(ar
->bmi
.cmd_buf
, 0, ar
->bmi
.max_data_size
+ header
);
195 ath6kl_dbg(ATH6KL_DBG_BMI
,
196 "bmi write memory: addr: 0x%x, len: %d\n", addr
, len
);
200 src
= &buf
[len
- len_remain
];
202 if (len_remain
< (ar
->bmi
.max_data_size
- header
)) {
203 if (len_remain
& 3) {
204 /* align it with 4 bytes */
205 len_remain
= len_remain
+
206 (4 - (len_remain
& 3));
207 memcpy(aligned_buf
, src
, len_remain
);
212 tx_len
= (ar
->bmi
.max_data_size
- header
);
216 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &cid
, sizeof(cid
));
217 offset
+= sizeof(cid
);
218 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &addr
, sizeof(addr
));
219 offset
+= sizeof(addr
);
220 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &tx_len
, sizeof(tx_len
));
221 offset
+= sizeof(tx_len
);
222 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), src
, tx_len
);
225 ret
= ath6kl_hif_bmi_write(ar
, ar
->bmi
.cmd_buf
, offset
);
227 ath6kl_err("Unable to write to the device: %d\n",
231 len_remain
-= tx_len
; addr
+= tx_len
;
237 int ath6kl_bmi_execute(struct ath6kl
*ar
, u32 addr
, u32
*param
)
239 u32 cid
= BMI_EXECUTE
;
244 if (ar
->bmi
.done_sent
) {
245 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
249 size
= sizeof(cid
) + sizeof(addr
) + sizeof(param
);
250 if (size
> ar
->bmi
.max_cmd_size
) {
254 memset(ar
->bmi
.cmd_buf
, 0, size
);
256 ath6kl_dbg(ATH6KL_DBG_BMI
, "bmi execute: addr: 0x%x, param: %d)\n",
260 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &cid
, sizeof(cid
));
261 offset
+= sizeof(cid
);
262 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &addr
, sizeof(addr
));
263 offset
+= sizeof(addr
);
264 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), param
, sizeof(*param
));
265 offset
+= sizeof(*param
);
267 ret
= ath6kl_hif_bmi_write(ar
, ar
->bmi
.cmd_buf
, offset
);
269 ath6kl_err("Unable to write to the device: %d\n", ret
);
273 ret
= ath6kl_hif_bmi_read(ar
, ar
->bmi
.cmd_buf
, sizeof(*param
));
275 ath6kl_err("Unable to read from the device: %d\n", ret
);
279 memcpy(param
, ar
->bmi
.cmd_buf
, sizeof(*param
));
284 int ath6kl_bmi_set_app_start(struct ath6kl
*ar
, u32 addr
)
286 u32 cid
= BMI_SET_APP_START
;
291 if (ar
->bmi
.done_sent
) {
292 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
296 size
= sizeof(cid
) + sizeof(addr
);
297 if (size
> ar
->bmi
.max_cmd_size
) {
301 memset(ar
->bmi
.cmd_buf
, 0, size
);
303 ath6kl_dbg(ATH6KL_DBG_BMI
, "bmi set app start: addr: 0x%x\n", addr
);
306 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &cid
, sizeof(cid
));
307 offset
+= sizeof(cid
);
308 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &addr
, sizeof(addr
));
309 offset
+= sizeof(addr
);
311 ret
= ath6kl_hif_bmi_write(ar
, ar
->bmi
.cmd_buf
, offset
);
313 ath6kl_err("Unable to write to the device: %d\n", ret
);
320 int ath6kl_bmi_reg_read(struct ath6kl
*ar
, u32 addr
, u32
*param
)
322 u32 cid
= BMI_READ_SOC_REGISTER
;
327 if (ar
->bmi
.done_sent
) {
328 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
332 size
= sizeof(cid
) + sizeof(addr
);
333 if (size
> ar
->bmi
.max_cmd_size
) {
337 memset(ar
->bmi
.cmd_buf
, 0, size
);
339 ath6kl_dbg(ATH6KL_DBG_BMI
, "bmi read SOC reg: addr: 0x%x\n", addr
);
342 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &cid
, sizeof(cid
));
343 offset
+= sizeof(cid
);
344 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &addr
, sizeof(addr
));
345 offset
+= sizeof(addr
);
347 ret
= ath6kl_hif_bmi_write(ar
, ar
->bmi
.cmd_buf
, offset
);
349 ath6kl_err("Unable to write to the device: %d\n", ret
);
353 ret
= ath6kl_hif_bmi_read(ar
, ar
->bmi
.cmd_buf
, sizeof(*param
));
355 ath6kl_err("Unable to read from the device: %d\n", ret
);
358 memcpy(param
, ar
->bmi
.cmd_buf
, sizeof(*param
));
363 int ath6kl_bmi_reg_write(struct ath6kl
*ar
, u32 addr
, u32 param
)
365 u32 cid
= BMI_WRITE_SOC_REGISTER
;
370 if (ar
->bmi
.done_sent
) {
371 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
375 size
= sizeof(cid
) + sizeof(addr
) + sizeof(param
);
376 if (size
> ar
->bmi
.max_cmd_size
) {
380 memset(ar
->bmi
.cmd_buf
, 0, size
);
382 ath6kl_dbg(ATH6KL_DBG_BMI
,
383 "bmi write SOC reg: addr: 0x%x, param: %d\n",
387 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &cid
, sizeof(cid
));
388 offset
+= sizeof(cid
);
389 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &addr
, sizeof(addr
));
390 offset
+= sizeof(addr
);
391 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), ¶m
, sizeof(param
));
392 offset
+= sizeof(param
);
394 ret
= ath6kl_hif_bmi_write(ar
, ar
->bmi
.cmd_buf
, offset
);
396 ath6kl_err("Unable to write to the device: %d\n", ret
);
403 int ath6kl_bmi_lz_data(struct ath6kl
*ar
, u8
*buf
, u32 len
)
405 u32 cid
= BMI_LZ_DATA
;
408 u32 len_remain
, tx_len
;
409 const u32 header
= sizeof(cid
) + sizeof(len
);
412 if (ar
->bmi
.done_sent
) {
413 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
417 size
= ar
->bmi
.max_data_size
+ header
;
418 if (size
> ar
->bmi
.max_cmd_size
) {
422 memset(ar
->bmi
.cmd_buf
, 0, size
);
424 ath6kl_dbg(ATH6KL_DBG_BMI
, "bmi send LZ data: len: %d)\n",
429 tx_len
= (len_remain
< (ar
->bmi
.max_data_size
- header
)) ?
430 len_remain
: (ar
->bmi
.max_data_size
- header
);
433 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &cid
, sizeof(cid
));
434 offset
+= sizeof(cid
);
435 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &tx_len
, sizeof(tx_len
));
436 offset
+= sizeof(tx_len
);
437 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &buf
[len
- len_remain
],
441 ret
= ath6kl_hif_bmi_write(ar
, ar
->bmi
.cmd_buf
, offset
);
443 ath6kl_err("Unable to write to the device: %d\n",
448 len_remain
-= tx_len
;
454 int ath6kl_bmi_lz_stream_start(struct ath6kl
*ar
, u32 addr
)
456 u32 cid
= BMI_LZ_STREAM_START
;
461 if (ar
->bmi
.done_sent
) {
462 ath6kl_err("bmi done sent already, cmd %d disallowed\n", cid
);
466 size
= sizeof(cid
) + sizeof(addr
);
467 if (size
> ar
->bmi
.max_cmd_size
) {
471 memset(ar
->bmi
.cmd_buf
, 0, size
);
473 ath6kl_dbg(ATH6KL_DBG_BMI
,
474 "bmi LZ stream start: addr: 0x%x)\n",
478 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &cid
, sizeof(cid
));
479 offset
+= sizeof(cid
);
480 memcpy(&(ar
->bmi
.cmd_buf
[offset
]), &addr
, sizeof(addr
));
481 offset
+= sizeof(addr
);
483 ret
= ath6kl_hif_bmi_write(ar
, ar
->bmi
.cmd_buf
, offset
);
485 ath6kl_err("Unable to start LZ stream to the device: %d\n",
493 int ath6kl_bmi_fast_download(struct ath6kl
*ar
, u32 addr
, u8
*buf
, u32 len
)
497 u32 last_word_offset
= len
& ~0x3;
498 u32 unaligned_bytes
= len
& 0x3;
500 ret
= ath6kl_bmi_lz_stream_start(ar
, addr
);
504 if (unaligned_bytes
) {
505 /* copy the last word into a zero padded buffer */
506 memcpy(&last_word
, &buf
[last_word_offset
], unaligned_bytes
);
509 ret
= ath6kl_bmi_lz_data(ar
, buf
, last_word_offset
);
514 ret
= ath6kl_bmi_lz_data(ar
, (u8
*)&last_word
, 4);
517 /* Close compressed stream and open a new (fake) one.
518 * This serves mainly to flush Target caches. */
519 ret
= ath6kl_bmi_lz_stream_start(ar
, 0x00);
524 void ath6kl_bmi_reset(struct ath6kl
*ar
)
526 ar
->bmi
.done_sent
= false;
529 int ath6kl_bmi_init(struct ath6kl
*ar
)
531 if (WARN_ON(ar
->bmi
.max_data_size
== 0))
534 /* cmd + addr + len + data_size */
535 ar
->bmi
.max_cmd_size
= ar
->bmi
.max_data_size
+ (sizeof(u32
) * 3);
537 ar
->bmi
.cmd_buf
= kzalloc(ar
->bmi
.max_cmd_size
, GFP_KERNEL
);
538 if (!ar
->bmi
.cmd_buf
)
544 void ath6kl_bmi_cleanup(struct ath6kl
*ar
)
546 kfree(ar
->bmi
.cmd_buf
);
547 ar
->bmi
.cmd_buf
= NULL
;